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 49static int ap_is_existing_file(char *pathname); 50 51static 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 81int 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 120int 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)tolower((int)filename[0]); 140 filename[1] = (char)tolower((int)filename[1]); 141 filename[2] = (char)tolower((int)filename[2]); 142 filename[3] = (char)tolower((int)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 strcat(filename, instance_str); 151 } 152 153 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 195struct 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 241cleanup: 242 acpi_os_close_file(file); 243 return (buffer); 244} 245