xref: /openbmc/linux/drivers/acpi/acpica/utstring.c (revision 7ae9fb1b7ecbb5d85d07857943f677fd1a559b18)
195857638SErik Schmauss // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
242f8fb75SBob Moore /*******************************************************************************
342f8fb75SBob Moore  *
442f8fb75SBob Moore  * Module Name: utstring - Common functions for strings and characters
542f8fb75SBob Moore  *
642f8fb75SBob Moore  ******************************************************************************/
742f8fb75SBob Moore 
842f8fb75SBob Moore #include <acpi/acpi.h>
942f8fb75SBob Moore #include "accommon.h"
1042f8fb75SBob Moore #include "acnamesp.h"
1142f8fb75SBob Moore 
1242f8fb75SBob Moore #define _COMPONENT          ACPI_UTILITIES
1342f8fb75SBob Moore ACPI_MODULE_NAME("utstring")
1442f8fb75SBob Moore 
1542f8fb75SBob Moore /*******************************************************************************
1642f8fb75SBob Moore  *
1742f8fb75SBob Moore  * FUNCTION:    acpi_ut_print_string
1842f8fb75SBob Moore  *
1942f8fb75SBob Moore  * PARAMETERS:  string          - Null terminated ASCII string
200fb3adf8SBob Moore  *              max_length      - Maximum output length. Used to constrain the
210fb3adf8SBob Moore  *                                length of strings during debug output only.
2242f8fb75SBob Moore  *
2342f8fb75SBob Moore  * RETURN:      None
2442f8fb75SBob Moore  *
2542f8fb75SBob Moore  * DESCRIPTION: Dump an ASCII string with support for ACPI-defined escape
2642f8fb75SBob Moore  *              sequences.
2742f8fb75SBob Moore  *
2842f8fb75SBob Moore  ******************************************************************************/
acpi_ut_print_string(char * string,u16 max_length)290fb3adf8SBob Moore void acpi_ut_print_string(char *string, u16 max_length)
3042f8fb75SBob Moore {
3142f8fb75SBob Moore 	u32 i;
3242f8fb75SBob Moore 
3342f8fb75SBob Moore 	if (!string) {
3442f8fb75SBob Moore 		acpi_os_printf("<\"NULL STRING PTR\">");
3542f8fb75SBob Moore 		return;
3642f8fb75SBob Moore 	}
3742f8fb75SBob Moore 
3842f8fb75SBob Moore 	acpi_os_printf("\"");
395d42b0faSDavid Binderman 	for (i = 0; (i < max_length) && string[i]; i++) {
4042f8fb75SBob Moore 
4142f8fb75SBob Moore 		/* Escape sequences */
4242f8fb75SBob Moore 
4342f8fb75SBob Moore 		switch (string[i]) {
4442f8fb75SBob Moore 		case 0x07:
451d1ea1b7SChao Guan 
4642f8fb75SBob Moore 			acpi_os_printf("\\a");	/* BELL */
4742f8fb75SBob Moore 			break;
4842f8fb75SBob Moore 
4942f8fb75SBob Moore 		case 0x08:
501d1ea1b7SChao Guan 
5142f8fb75SBob Moore 			acpi_os_printf("\\b");	/* BACKSPACE */
5242f8fb75SBob Moore 			break;
5342f8fb75SBob Moore 
5442f8fb75SBob Moore 		case 0x0C:
551d1ea1b7SChao Guan 
5642f8fb75SBob Moore 			acpi_os_printf("\\f");	/* FORMFEED */
5742f8fb75SBob Moore 			break;
5842f8fb75SBob Moore 
5942f8fb75SBob Moore 		case 0x0A:
601d1ea1b7SChao Guan 
6142f8fb75SBob Moore 			acpi_os_printf("\\n");	/* LINEFEED */
6242f8fb75SBob Moore 			break;
6342f8fb75SBob Moore 
6442f8fb75SBob Moore 		case 0x0D:
651d1ea1b7SChao Guan 
6642f8fb75SBob Moore 			acpi_os_printf("\\r");	/* CARRIAGE RETURN */
6742f8fb75SBob Moore 			break;
6842f8fb75SBob Moore 
6942f8fb75SBob Moore 		case 0x09:
701d1ea1b7SChao Guan 
7142f8fb75SBob Moore 			acpi_os_printf("\\t");	/* HORIZONTAL TAB */
7242f8fb75SBob Moore 			break;
7342f8fb75SBob Moore 
7442f8fb75SBob Moore 		case 0x0B:
751d1ea1b7SChao Guan 
7642f8fb75SBob Moore 			acpi_os_printf("\\v");	/* VERTICAL TAB */
7742f8fb75SBob Moore 			break;
7842f8fb75SBob Moore 
7942f8fb75SBob Moore 		case '\'':	/* Single Quote */
8042f8fb75SBob Moore 		case '\"':	/* Double Quote */
8142f8fb75SBob Moore 		case '\\':	/* Backslash */
821d1ea1b7SChao Guan 
8342f8fb75SBob Moore 			acpi_os_printf("\\%c", (int)string[i]);
8442f8fb75SBob Moore 			break;
8542f8fb75SBob Moore 
8642f8fb75SBob Moore 		default:
8742f8fb75SBob Moore 
8842f8fb75SBob Moore 			/* Check for printable character or hex escape */
8942f8fb75SBob Moore 
904fa4616eSBob Moore 			if (isprint((int)string[i])) {
9142f8fb75SBob Moore 				/* This is a normal character */
9242f8fb75SBob Moore 
9342f8fb75SBob Moore 				acpi_os_printf("%c", (int)string[i]);
9442f8fb75SBob Moore 			} else {
9542f8fb75SBob Moore 				/* All others will be Hex escapes */
9642f8fb75SBob Moore 
9742f8fb75SBob Moore 				acpi_os_printf("\\x%2.2X", (s32)string[i]);
9842f8fb75SBob Moore 			}
9942f8fb75SBob Moore 			break;
10042f8fb75SBob Moore 		}
10142f8fb75SBob Moore 	}
1021fad8738SBob Moore 
10342f8fb75SBob Moore 	acpi_os_printf("\"");
10442f8fb75SBob Moore 
10542f8fb75SBob Moore 	if (i == max_length && string[i]) {
10642f8fb75SBob Moore 		acpi_os_printf("...");
10742f8fb75SBob Moore 	}
10842f8fb75SBob Moore }
10942f8fb75SBob Moore 
11042f8fb75SBob Moore /*******************************************************************************
11142f8fb75SBob Moore  *
11242f8fb75SBob Moore  * FUNCTION:    acpi_ut_repair_name
11342f8fb75SBob Moore  *
11442f8fb75SBob Moore  * PARAMETERS:  name            - The ACPI name to be repaired
11542f8fb75SBob Moore  *
11642f8fb75SBob Moore  * RETURN:      Repaired version of the name
11742f8fb75SBob Moore  *
11842f8fb75SBob Moore  * DESCRIPTION: Repair an ACPI name: Change invalid characters to '*' and
11942f8fb75SBob Moore  *              return the new name. NOTE: the Name parameter must reside in
12042f8fb75SBob Moore  *              read/write memory, cannot be a const.
12142f8fb75SBob Moore  *
12242f8fb75SBob Moore  * An ACPI Name must consist of valid ACPI characters. We will repair the name
12342f8fb75SBob Moore  * if necessary because we don't want to abort because of this, but we want
12442f8fb75SBob Moore  * all namespace names to be printable. A warning message is appropriate.
12542f8fb75SBob Moore  *
12642f8fb75SBob Moore  * This issue came up because there are in fact machines that exhibit
12742f8fb75SBob Moore  * this problem, and we want to be able to enable ACPI support for them,
12842f8fb75SBob Moore  * even though there are a few bad names.
12942f8fb75SBob Moore  *
13042f8fb75SBob Moore  ******************************************************************************/
13142f8fb75SBob Moore 
acpi_ut_repair_name(char * name)13242f8fb75SBob Moore void acpi_ut_repair_name(char *name)
13342f8fb75SBob Moore {
13442f8fb75SBob Moore 	u32 i;
13542f8fb75SBob Moore 	u8 found_bad_char = FALSE;
13642f8fb75SBob Moore 	u32 original_name;
13742f8fb75SBob Moore 
13842f8fb75SBob Moore 	ACPI_FUNCTION_NAME(ut_repair_name);
13942f8fb75SBob Moore 
14022472353SBob Moore 	/*
14122472353SBob Moore 	 * Special case for the root node. This can happen if we get an
14222472353SBob Moore 	 * error during the execution of module-level code.
14322472353SBob Moore 	 */
1445599fb69SBob Moore 	if (ACPI_COMPARE_NAMESEG(name, ACPI_ROOT_PATHNAME)) {
14522472353SBob Moore 		return;
14622472353SBob Moore 	}
14722472353SBob Moore 
148*183f0a09SBob Moore 	ACPI_COPY_NAMESEG(&original_name, &name[0]);
14942f8fb75SBob Moore 
15042f8fb75SBob Moore 	/* Check each character in the name */
15142f8fb75SBob Moore 
15232786755SBob Moore 	for (i = 0; i < ACPI_NAMESEG_SIZE; i++) {
1536a0df32cSBob Moore 		if (acpi_ut_valid_name_char(name[i], i)) {
15442f8fb75SBob Moore 			continue;
15542f8fb75SBob Moore 		}
15642f8fb75SBob Moore 
15742f8fb75SBob Moore 		/*
15842f8fb75SBob Moore 		 * Replace a bad character with something printable, yet technically
159*183f0a09SBob Moore 		 * "odd". This prevents any collisions with existing "good"
16042f8fb75SBob Moore 		 * names in the namespace.
16142f8fb75SBob Moore 		 */
162*183f0a09SBob Moore 		name[i] = '_';
16342f8fb75SBob Moore 		found_bad_char = TRUE;
16442f8fb75SBob Moore 	}
16542f8fb75SBob Moore 
16642f8fb75SBob Moore 	if (found_bad_char) {
16742f8fb75SBob Moore 
16842f8fb75SBob Moore 		/* Report warning only if in strict mode or debug mode */
16942f8fb75SBob Moore 
17042f8fb75SBob Moore 		if (!acpi_gbl_enable_interpreter_slack) {
17142f8fb75SBob Moore 			ACPI_WARNING((AE_INFO,
172*183f0a09SBob Moore 				      "Invalid character(s) in name (0x%.8X) %p, repaired: [%4.4s]",
173*183f0a09SBob Moore 				      original_name, name, &name[0]));
17442f8fb75SBob Moore 		} else {
17542f8fb75SBob Moore 			ACPI_DEBUG_PRINT((ACPI_DB_INFO,
17642f8fb75SBob Moore 					  "Invalid character(s) in name (0x%.8X), repaired: [%4.4s]",
17742f8fb75SBob Moore 					  original_name, name));
17842f8fb75SBob Moore 		}
17942f8fb75SBob Moore 	}
18042f8fb75SBob Moore }
18142f8fb75SBob Moore 
18242f8fb75SBob Moore #if defined ACPI_ASL_COMPILER || defined ACPI_EXEC_APP
18342f8fb75SBob Moore /*******************************************************************************
18442f8fb75SBob Moore  *
18542f8fb75SBob Moore  * FUNCTION:    ut_convert_backslashes
18642f8fb75SBob Moore  *
18742f8fb75SBob Moore  * PARAMETERS:  pathname        - File pathname string to be converted
18842f8fb75SBob Moore  *
18942f8fb75SBob Moore  * RETURN:      Modifies the input Pathname
19042f8fb75SBob Moore  *
19142f8fb75SBob Moore  * DESCRIPTION: Convert all backslashes (0x5C) to forward slashes (0x2F) within
19242f8fb75SBob Moore  *              the entire input file pathname string.
19342f8fb75SBob Moore  *
19442f8fb75SBob Moore  ******************************************************************************/
19542f8fb75SBob Moore 
ut_convert_backslashes(char * pathname)19642f8fb75SBob Moore void ut_convert_backslashes(char *pathname)
19742f8fb75SBob Moore {
19842f8fb75SBob Moore 
19942f8fb75SBob Moore 	if (!pathname) {
20042f8fb75SBob Moore 		return;
20142f8fb75SBob Moore 	}
20242f8fb75SBob Moore 
20342f8fb75SBob Moore 	while (*pathname) {
20442f8fb75SBob Moore 		if (*pathname == '\\') {
20542f8fb75SBob Moore 			*pathname = '/';
20642f8fb75SBob Moore 		}
20742f8fb75SBob Moore 
20842f8fb75SBob Moore 		pathname++;
20942f8fb75SBob Moore 	}
21042f8fb75SBob Moore }
21142f8fb75SBob Moore #endif
212