1 // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2 /******************************************************************************
3  *
4  * Module Name: apfiles - File-related functions for acpidump utility
5  *
6  * Copyright (C) 2000 - 2023, Intel Corp.
7  *
8  *****************************************************************************/
9 
10 #include "acpidump.h"
11 
12 /* Local prototypes */
13 
14 static int ap_is_existing_file(char *pathname);
15 
16 /******************************************************************************
17  *
18  * FUNCTION:    ap_is_existing_file
19  *
20  * PARAMETERS:  pathname            - Output filename
21  *
22  * RETURN:      0 on success
23  *
24  * DESCRIPTION: Query for file overwrite if it already exists.
25  *
26  ******************************************************************************/
27 
ap_is_existing_file(char * pathname)28 static int ap_is_existing_file(char *pathname)
29 {
30 #if !defined(_GNU_EFI) && !defined(_EDK2_EFI)
31 	struct stat stat_info;
32 	int in_char;
33 
34 	if (!stat(pathname, &stat_info)) {
35 		fprintf(stderr,
36 			"Target path already exists, overwrite? [y|n] ");
37 
38 		in_char = fgetc(stdin);
39 		if (in_char == '\n') {
40 			in_char = fgetc(stdin);
41 		}
42 
43 		if (in_char != 'y' && in_char != 'Y') {
44 			return (-1);
45 		}
46 	}
47 #endif
48 
49 	return (0);
50 }
51 
52 /******************************************************************************
53  *
54  * FUNCTION:    ap_open_output_file
55  *
56  * PARAMETERS:  pathname            - Output filename
57  *
58  * RETURN:      Open file handle
59  *
60  * DESCRIPTION: Open a text output file for acpidump. Checks if file already
61  *              exists.
62  *
63  ******************************************************************************/
64 
ap_open_output_file(char * pathname)65 int ap_open_output_file(char *pathname)
66 {
67 	ACPI_FILE file;
68 
69 	/* If file exists, prompt for overwrite */
70 
71 	if (ap_is_existing_file(pathname) != 0) {
72 		return (-1);
73 	}
74 
75 	/* Point stdout to the file */
76 
77 	file = fopen(pathname, "w");
78 	if (!file) {
79 		fprintf(stderr, "Could not open output file: %s\n", pathname);
80 		return (-1);
81 	}
82 
83 	/* Save the file and path */
84 
85 	gbl_output_file = file;
86 	gbl_output_filename = pathname;
87 	return (0);
88 }
89 
90 /******************************************************************************
91  *
92  * FUNCTION:    ap_write_to_binary_file
93  *
94  * PARAMETERS:  table               - ACPI table to be written
95  *              instance            - ACPI table instance no. to be written
96  *
97  * RETURN:      Status
98  *
99  * DESCRIPTION: Write an ACPI table to a binary file. Builds the output
100  *              filename from the table signature.
101  *
102  ******************************************************************************/
103 
ap_write_to_binary_file(struct acpi_table_header * table,u32 instance)104 int ap_write_to_binary_file(struct acpi_table_header *table, u32 instance)
105 {
106 	char filename[ACPI_NAMESEG_SIZE + 16];
107 	char instance_str[16];
108 	ACPI_FILE file;
109 	acpi_size actual;
110 	u32 table_length;
111 
112 	/* Obtain table length */
113 
114 	table_length = ap_get_table_length(table);
115 
116 	/* Construct lower-case filename from the table local signature */
117 
118 	if (ACPI_VALIDATE_RSDP_SIG(table->signature)) {
119 		ACPI_COPY_NAMESEG(filename, ACPI_RSDP_NAME);
120 	} else {
121 		ACPI_COPY_NAMESEG(filename, table->signature);
122 	}
123 
124 	filename[0] = (char)tolower((int)filename[0]);
125 	filename[1] = (char)tolower((int)filename[1]);
126 	filename[2] = (char)tolower((int)filename[2]);
127 	filename[3] = (char)tolower((int)filename[3]);
128 	filename[ACPI_NAMESEG_SIZE] = 0;
129 
130 	/* Handle multiple SSDts - create different filenames for each */
131 
132 	if (instance > 0) {
133 		snprintf(instance_str, sizeof(instance_str), "%u", instance);
134 		strcat(filename, instance_str);
135 	}
136 
137 	strcat(filename, FILE_SUFFIX_BINARY_TABLE);
138 
139 	if (gbl_verbose_mode) {
140 		fprintf(stderr,
141 			"Writing [%4.4s] to binary file: %s 0x%X (%u) bytes\n",
142 			table->signature, filename, table->length,
143 			table->length);
144 	}
145 
146 	/* Open the file and dump the entire table in binary mode */
147 
148 	file = fopen(filename, "wb");
149 	if (!file) {
150 		fprintf(stderr, "Could not open output file: %s\n", filename);
151 		return (-1);
152 	}
153 
154 	actual = fwrite(table, 1, table_length, file);
155 	if (actual != table_length) {
156 		fprintf(stderr, "Error writing binary output file: %s\n",
157 			filename);
158 		fclose(file);
159 		return (-1);
160 	}
161 
162 	fclose(file);
163 	return (0);
164 }
165 
166 /******************************************************************************
167  *
168  * FUNCTION:    ap_get_table_from_file
169  *
170  * PARAMETERS:  pathname            - File containing the binary ACPI table
171  *              out_file_size       - Where the file size is returned
172  *
173  * RETURN:      Buffer containing the ACPI table. NULL on error.
174  *
175  * DESCRIPTION: Open a file and read it entirely into a new buffer
176  *
177  ******************************************************************************/
178 
ap_get_table_from_file(char * pathname,u32 * out_file_size)179 struct acpi_table_header *ap_get_table_from_file(char *pathname,
180 						 u32 *out_file_size)
181 {
182 	struct acpi_table_header *buffer = NULL;
183 	ACPI_FILE file;
184 	u32 file_size;
185 	acpi_size actual;
186 
187 	/* Must use binary mode */
188 
189 	file = fopen(pathname, "rb");
190 	if (!file) {
191 		fprintf(stderr, "Could not open input file: %s\n", pathname);
192 		return (NULL);
193 	}
194 
195 	/* Need file size to allocate a buffer */
196 
197 	file_size = cm_get_file_size(file);
198 	if (file_size == ACPI_UINT32_MAX) {
199 		fprintf(stderr,
200 			"Could not get input file size: %s\n", pathname);
201 		goto cleanup;
202 	}
203 
204 	/* Allocate a buffer for the entire file */
205 
206 	buffer = ACPI_ALLOCATE_ZEROED(file_size);
207 	if (!buffer) {
208 		fprintf(stderr,
209 			"Could not allocate file buffer of size: %u\n",
210 			file_size);
211 		goto cleanup;
212 	}
213 
214 	/* Read the entire file */
215 
216 	actual = fread(buffer, 1, file_size, file);
217 	if (actual != file_size) {
218 		fprintf(stderr, "Could not read input file: %s\n", pathname);
219 		ACPI_FREE(buffer);
220 		buffer = NULL;
221 		goto cleanup;
222 	}
223 
224 	*out_file_size = file_size;
225 
226 cleanup:
227 	fclose(file);
228 	return (buffer);
229 }
230