linux/tools/power/acpi/tools/acpidump/apfiles.c
<<
>>
Prefs
   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