Add the rt linux 4.1.3-rt3 as base
[kvmfornfv.git] / kernel / tools / power / acpi / tools / acpidump / apfiles.c
1 /******************************************************************************
2  *
3  * Module Name: apfiles - File-related functions for acpidump utility
4  *
5  *****************************************************************************/
6
7 /*
8  * Copyright (C) 2000 - 2015, Intel Corp.
9  * All rights reserved.
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions, and the following disclaimer,
16  *    without modification.
17  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18  *    substantially similar to the "NO WARRANTY" disclaimer below
19  *    ("Disclaimer") and any redistribution must be conditioned upon
20  *    including a substantially similar Disclaimer requirement for further
21  *    binary redistribution.
22  * 3. Neither the names of the above-listed copyright holders nor the names
23  *    of any contributors may be used to endorse or promote products derived
24  *    from this software without specific prior written permission.
25  *
26  * Alternatively, this software may be distributed under the terms of the
27  * GNU General Public License ("GPL") version 2 as published by the Free
28  * Software Foundation.
29  *
30  * NO WARRANTY
31  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41  * POSSIBILITY OF SUCH DAMAGES.
42  */
43
44 #include "acpidump.h"
45 #include "acapps.h"
46
47 /* Local prototypes */
48
49 static int ap_is_existing_file(char *pathname);
50
51 static int ap_is_existing_file(char *pathname)
52 {
53 #ifndef _GNU_EFI
54         struct stat stat_info;
55
56         if (!stat(pathname, &stat_info)) {
57                 acpi_log_error("Target path already exists, overwrite? [y|n] ");
58
59                 if (getchar() != 'y') {
60                         return (-1);
61                 }
62         }
63 #endif
64
65         return 0;
66 }
67
68 /******************************************************************************
69  *
70  * FUNCTION:    ap_open_output_file
71  *
72  * PARAMETERS:  pathname            - Output filename
73  *
74  * RETURN:      Open file handle
75  *
76  * DESCRIPTION: Open a text output file for acpidump. Checks if file already
77  *              exists.
78  *
79  ******************************************************************************/
80
81 int ap_open_output_file(char *pathname)
82 {
83         ACPI_FILE file;
84
85         /* If file exists, prompt for overwrite */
86
87         if (ap_is_existing_file(pathname) != 0) {
88                 return (-1);
89         }
90
91         /* Point stdout to the file */
92
93         file = acpi_os_open_file(pathname, ACPI_FILE_WRITING);
94         if (!file) {
95                 acpi_log_error("Could not open output file: %s\n", pathname);
96                 return (-1);
97         }
98
99         /* Save the file and path */
100
101         gbl_output_file = file;
102         gbl_output_filename = pathname;
103         return (0);
104 }
105
106 /******************************************************************************
107  *
108  * FUNCTION:    ap_write_to_binary_file
109  *
110  * PARAMETERS:  table               - ACPI table to be written
111  *              instance            - ACPI table instance no. to be written
112  *
113  * RETURN:      Status
114  *
115  * DESCRIPTION: Write an ACPI table to a binary file. Builds the output
116  *              filename from the table signature.
117  *
118  ******************************************************************************/
119
120 int ap_write_to_binary_file(struct acpi_table_header *table, u32 instance)
121 {
122         char filename[ACPI_NAME_SIZE + 16];
123         char instance_str[16];
124         ACPI_FILE file;
125         size_t actual;
126         u32 table_length;
127
128         /* Obtain table length */
129
130         table_length = ap_get_table_length(table);
131
132         /* Construct lower-case filename from the table local signature */
133
134         if (ACPI_VALIDATE_RSDP_SIG(table->signature)) {
135                 ACPI_MOVE_NAME(filename, ACPI_RSDP_NAME);
136         } else {
137                 ACPI_MOVE_NAME(filename, table->signature);
138         }
139         filename[0] = (char)ACPI_TOLOWER(filename[0]);
140         filename[1] = (char)ACPI_TOLOWER(filename[1]);
141         filename[2] = (char)ACPI_TOLOWER(filename[2]);
142         filename[3] = (char)ACPI_TOLOWER(filename[3]);
143         filename[ACPI_NAME_SIZE] = 0;
144
145         /* Handle multiple SSDts - create different filenames for each */
146
147         if (instance > 0) {
148                 acpi_ut_snprintf(instance_str, sizeof(instance_str), "%u",
149                                  instance);
150                 ACPI_STRCAT(filename, instance_str);
151         }
152
153         ACPI_STRCAT(filename, ACPI_TABLE_FILE_SUFFIX);
154
155         if (gbl_verbose_mode) {
156                 acpi_log_error
157                     ("Writing [%4.4s] to binary file: %s 0x%X (%u) bytes\n",
158                      table->signature, filename, table->length, table->length);
159         }
160
161         /* Open the file and dump the entire table in binary mode */
162
163         file = acpi_os_open_file(filename,
164                                  ACPI_FILE_WRITING | ACPI_FILE_BINARY);
165         if (!file) {
166                 acpi_log_error("Could not open output file: %s\n", filename);
167                 return (-1);
168         }
169
170         actual = acpi_os_write_file(file, table, 1, table_length);
171         if (actual != table_length) {
172                 acpi_log_error("Error writing binary output file: %s\n",
173                                filename);
174                 acpi_os_close_file(file);
175                 return (-1);
176         }
177
178         acpi_os_close_file(file);
179         return (0);
180 }
181
182 /******************************************************************************
183  *
184  * FUNCTION:    ap_get_table_from_file
185  *
186  * PARAMETERS:  pathname            - File containing the binary ACPI table
187  *              out_file_size       - Where the file size is returned
188  *
189  * RETURN:      Buffer containing the ACPI table. NULL on error.
190  *
191  * DESCRIPTION: Open a file and read it entirely into a new buffer
192  *
193  ******************************************************************************/
194
195 struct acpi_table_header *ap_get_table_from_file(char *pathname,
196                                                  u32 *out_file_size)
197 {
198         struct acpi_table_header *buffer = NULL;
199         ACPI_FILE file;
200         u32 file_size;
201         size_t actual;
202
203         /* Must use binary mode */
204
205         file =
206             acpi_os_open_file(pathname, ACPI_FILE_READING | ACPI_FILE_BINARY);
207         if (!file) {
208                 acpi_log_error("Could not open input file: %s\n", pathname);
209                 return (NULL);
210         }
211
212         /* Need file size to allocate a buffer */
213
214         file_size = cm_get_file_size(file);
215         if (file_size == ACPI_UINT32_MAX) {
216                 acpi_log_error("Could not get input file size: %s\n", pathname);
217                 goto cleanup;
218         }
219
220         /* Allocate a buffer for the entire file */
221
222         buffer = ACPI_ALLOCATE_ZEROED(file_size);
223         if (!buffer) {
224                 acpi_log_error("Could not allocate file buffer of size: %u\n",
225                                file_size);
226                 goto cleanup;
227         }
228
229         /* Read the entire file */
230
231         actual = acpi_os_read_file(file, buffer, 1, file_size);
232         if (actual != file_size) {
233                 acpi_log_error("Could not read input file: %s\n", pathname);
234                 ACPI_FREE(buffer);
235                 buffer = NULL;
236                 goto cleanup;
237         }
238
239         *out_file_size = file_size;
240
241 cleanup:
242         acpi_os_close_file(file);
243         return (buffer);
244 }