8b86f21f45
Add support to walk through PSP L1, PSP L2, BIOS L1, BIOS L2 directories and list the entries present in them. Accommodate both recovery A/B layout and normal layout. This is required to identify the location and size of each entries in the finally built amdfw.rom. This in turn can be used to perform any platform specific verification on the relevant components. BUG=None TEST=Build and list the contents of AMDFW binary. /usr/bin/amdfwread --ro-list /build/skyrim/firmware/image-skyrim.bin Table: FW Offset Size PSPL1: Dir 0x00d97000 +-->PSPL1: 0x48 0x00d98000 0x00001000 +-->PSPL2: Dir 0x00c30000 +-->PSPL2: 0x00 0x00c31000 0x00000440 +-->PSPL2: 0x01 0x00c31500 0x00007580 +-->PSPL2: 0x02 0x00c38b00 0x00019470 +-->PSPL2: 0x08 0x00c52000 0x0001f560 +-->PSPL2: 0x09 0x00c71600 0x00000440 +-->PSPL2: 0x0b 0x430000041(Soft-fuse) +-->PSPL2: 0x0c 0x00c71b00 0x00023100 +-->PSPL2: 0x12 0x00c94c00 0x00015890 +-->PSPL2: 0x13 0x00caa500 0x000021c0 +-->PSPL2: 0x20 0x00cac700 0x00000640 +-->PSPL2: 0x21 0x00cace00 0x00000030 +-->PSPL2: 0x22 0x00cad000 0x00001000 +-->PSPL2: 0x24 0x00cae000 0x00003b60 +-->PSPL2: 0x28 0x00cb1c00 0x00022890 +-->PSPL2: 0x2d 0x00cd4500 0x00003100 +-->PSPL2: 0x30 0x00cd7600 0x0006b550 +-->PSPL2: 0x3a 0x00d42c00 0x000006d0 +-->PSPL2: 0x3c 0x00d43300 0x000018c0 +-->PSPL2: 0x44 0x00d44c00 0x00006610 +-->PSPL2: 0x45 0x00d4b300 0x00001c70 +-->PSPL2: 0x50 0x00d4d000 0x00001a00 +-->PSPL2: 0x51 0x00d4ea00 0x00001020 +-->PSPL2: 0x52 0x00d4fb00 0x00010180 +-->PSPL2: 0x55 0x00d5fd00 0x00000600 +-->PSPL2: 0x5a 0x00d60300 0x00000570 +-->PSPL2: 0x5c 0x00d60900 0x00000b20 +-->PSPL2: 0x71 0x00d61500 0x00024710 +-->PSPL2: 0x73 0x00d85d00 0x00010640 +-->PSPL2: 0x8d 0x00d96400 0x00000030 +-->PSPL2: 0x49 0x00d99000 0x00001000 +-->BIOSL2: Dir 0x00d99000 +-->BIOSL2: 0x60 0x00d9a000 0x00009924 +-->BIOSL2: 0x68 0x00da4000 0x00009924 +-->BIOSL2: 0x61 0x2001000(DRAM-Address) +-->BIOSL2: 0x62 0x00dada00 0x00010000 +-->BIOSL2: 0x63 0x00000000 0x0001e000 +-->BIOSL2: 0x64 0x00db4200 0x00006310 +-->BIOSL2: 0x65 0x00dba600 0x000004e0 +-->BIOSL2: 0x64 0x00dbab00 0x00006180 +-->BIOSL2: 0x65 0x00dc0d00 0x00000250 +-->BIOSL2: 0x6b 0x201f000(DRAM-Address) +-->PSPL1: 0x4a 0x00d98000 0x00001000 Change-Id: Ia1b8f1a2b9bc7dc6925a305cdff1442aaff182cd Signed-off-by: Karthikeyan Ramasubramanian <kramasub@google.com> Reviewed-on: https://review.coreboot.org/c/coreboot/+/66761 Reviewed-by: Raul Rangel <rrangel@chromium.org> Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
511 lines
14 KiB
C
511 lines
14 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
#include <getopt.h>
|
|
#include <stddef.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include "amdfwtool.h"
|
|
|
|
/* An address can be relative to the image/file start but it can also be the address when
|
|
* the image is mapped at 0xff000000. Used to ensure that we only attempt to read within
|
|
* the limits of the file. */
|
|
#define SPI_ROM_BASE 0xff000000
|
|
#define FILE_REL_MASK 0xffffff
|
|
|
|
#define ERR(...) fprintf(stderr, __VA_ARGS__)
|
|
|
|
/* Possible locations for the header */
|
|
const uint32_t fw_header_offsets[] = {
|
|
0xfa0000,
|
|
0xe20000,
|
|
0xc20000,
|
|
0x820000,
|
|
0x020000,
|
|
};
|
|
|
|
/* Converts addresses to be relative to the start of the file */
|
|
static uint64_t relative_offset(uint32_t header_offset, uint64_t addr, uint64_t mode)
|
|
{
|
|
switch (mode) {
|
|
/* Since this utility operates on the BIOS file, physical address is converted
|
|
relative to the start of the BIOS file. */
|
|
case AMD_ADDR_PHYSICAL:
|
|
if (addr < SPI_ROM_BASE || addr > (SPI_ROM_BASE + FILE_REL_MASK)) {
|
|
ERR("Invalid address(%lx) or mode(%lx)\n", addr, mode);
|
|
/* TODO: fix amdfwtool to program the right address/mode. In guybrush,
|
|
* lots of addresses are marked as physical, but they are relative to
|
|
* BIOS. Until that is fixed, just leave an error message. */
|
|
// exit(1);
|
|
}
|
|
return addr & FILE_REL_MASK;
|
|
|
|
case AMD_ADDR_REL_BIOS:
|
|
if (addr > FILE_REL_MASK) {
|
|
ERR("Invalid address(%lx) or mode(%lx)\n", addr, mode);
|
|
exit(1);
|
|
}
|
|
return addr & FILE_REL_MASK;
|
|
|
|
case AMD_ADDR_REL_TAB:
|
|
return addr + header_offset;
|
|
|
|
default:
|
|
ERR("Unsupported mode %lu\n", mode);
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
static int read_header(FILE *fw, uint32_t offset, void *header, size_t header_size)
|
|
{
|
|
if (fseek(fw, offset, SEEK_SET) != 0) {
|
|
ERR("Failed to seek to file offset 0x%x\n", offset);
|
|
return 1;
|
|
}
|
|
|
|
if (fread(header, header_size, 1, fw) != 1) {
|
|
ERR("Failed to read header at 0x%x\n", offset);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int read_fw_header(FILE *fw, uint32_t offset, embedded_firmware *fw_header)
|
|
{
|
|
if (read_header(fw, offset, fw_header, sizeof(embedded_firmware))) {
|
|
ERR("Failed to read fw header at 0x%x\n", offset);
|
|
return 1;
|
|
}
|
|
|
|
return fw_header->signature != EMBEDDED_FW_SIGNATURE;
|
|
}
|
|
|
|
static int read_psp_directory(FILE *fw, uint32_t offset, uint32_t expected_cookie,
|
|
psp_directory_header *header, psp_directory_entry **entries,
|
|
size_t *num_entries)
|
|
{
|
|
offset &= FILE_REL_MASK;
|
|
|
|
if (read_header(fw, offset, header, sizeof(psp_directory_header))) {
|
|
ERR("Failed to read PSP header\n");
|
|
return 1;
|
|
}
|
|
|
|
/* Ensure that we have a PSP directory */
|
|
if (header->cookie != expected_cookie) {
|
|
ERR("Invalid PSP header cookie value found: 0x%x, expected: 0x%x\n",
|
|
header->cookie, expected_cookie);
|
|
return 1;
|
|
}
|
|
|
|
/* Read the entries */
|
|
*num_entries = header->num_entries;
|
|
*entries = malloc(sizeof(psp_directory_entry) * header->num_entries);
|
|
if (fread(*entries, sizeof(psp_directory_entry), header->num_entries, fw)
|
|
!= header->num_entries) {
|
|
ERR("Failed to read %d PSP entries\n", header->num_entries);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int read_ish_directory(FILE *fw, uint32_t offset, ish_directory_table *table)
|
|
{
|
|
return read_header(fw, offset & FILE_REL_MASK, table, sizeof(*table));
|
|
}
|
|
|
|
static int read_bios_directory(FILE *fw, uint32_t offset, uint32_t expected_cookie,
|
|
bios_directory_hdr *header, bios_directory_entry **entries,
|
|
size_t *num_entries)
|
|
{
|
|
offset &= FILE_REL_MASK;
|
|
|
|
if (read_header(fw, offset, header, sizeof(bios_directory_hdr))) {
|
|
ERR("Failed to read BIOS header\n");
|
|
return 1;
|
|
}
|
|
|
|
/* Ensure that we have a BIOS directory */
|
|
if (header->cookie != expected_cookie) {
|
|
ERR("Invalid BIOS header cookie value found: 0x%x, expected: 0x%x\n",
|
|
header->cookie, expected_cookie);
|
|
return 1;
|
|
}
|
|
|
|
/* Read the entries */
|
|
*num_entries = header->num_entries;
|
|
*entries = malloc(sizeof(bios_directory_entry) * header->num_entries);
|
|
if (fread(*entries, sizeof(bios_directory_entry), header->num_entries, fw)
|
|
!= header->num_entries) {
|
|
ERR("Failed to read %d BIOS entries\n", header->num_entries);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int read_soft_fuse(FILE *fw, const embedded_firmware *fw_header)
|
|
{
|
|
psp_directory_entry *current_entries = NULL;
|
|
size_t num_current_entries = 0;
|
|
|
|
uint32_t psp_offset = 0;
|
|
/* 0xffffffff indicates that the offset is in new_psp_directory */
|
|
if (fw_header->psp_directory != 0xffffffff)
|
|
psp_offset = fw_header->psp_directory;
|
|
else
|
|
psp_offset = fw_header->new_psp_directory;
|
|
|
|
psp_directory_header header;
|
|
if (read_psp_directory(fw, psp_offset, PSP_COOKIE, &header,
|
|
¤t_entries, &num_current_entries) != 0)
|
|
return 1;
|
|
|
|
while (1) {
|
|
uint32_t l2_dir_offset = 0;
|
|
uint32_t ish_dir_offset;
|
|
ish_directory_table ish_dir;
|
|
|
|
for (size_t i = 0; i < num_current_entries; i++) {
|
|
uint32_t type = current_entries[i].type;
|
|
uint64_t mode = current_entries[i].address_mode;
|
|
uint64_t addr = current_entries[i].addr;
|
|
uint64_t fuse;
|
|
|
|
switch (type) {
|
|
case AMD_PSP_FUSE_CHAIN:
|
|
fuse = mode << 62 | addr;
|
|
|
|
printf("Soft-fuse:0x%lx\n", fuse);
|
|
free(current_entries);
|
|
return 0;
|
|
|
|
case AMD_FW_L2_PTR:
|
|
/* There's a second level PSP directory to read */
|
|
if (l2_dir_offset != 0) {
|
|
ERR("Duplicate PSP L2 Entry, prior offset: %08x\n",
|
|
l2_dir_offset);
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
l2_dir_offset = relative_offset(psp_offset, addr, mode);
|
|
break;
|
|
|
|
case AMD_FW_RECOVERYAB_A:
|
|
if (l2_dir_offset != 0) {
|
|
ERR("Duplicate PSP L2 Entry, prior offset: %08x\n",
|
|
l2_dir_offset);
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
ish_dir_offset = relative_offset(psp_offset, addr, mode);
|
|
if (read_ish_directory(fw, ish_dir_offset, &ish_dir) != 0) {
|
|
ERR("Error reading ISH directory\n");
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
l2_dir_offset = ish_dir.pl2_location;
|
|
break;
|
|
|
|
default:
|
|
/* No-op, continue to the next entry. */
|
|
break;
|
|
}
|
|
}
|
|
|
|
free(current_entries);
|
|
|
|
/* Didn't find an L2 PSP directory so we can stop */
|
|
if (l2_dir_offset == 0)
|
|
break;
|
|
|
|
/* Read the L2 PSP directory */
|
|
if (read_psp_directory(fw, l2_dir_offset, PSPL2_COOKIE, &header,
|
|
¤t_entries, &num_current_entries) != 0)
|
|
break;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#define MAX_NUM_LEVELS 10
|
|
#define MAX_INDENT_PER_LEVEL 4
|
|
#define MAX_INDENTATION_LEN (MAX_NUM_LEVELS * MAX_INDENT_PER_LEVEL + 1)
|
|
static void do_indentation_string(char *dest, uint8_t level)
|
|
{
|
|
for (uint8_t i = 0; i < level && i < MAX_NUM_LEVELS; i++)
|
|
strcat(dest, " ");
|
|
strcat(dest, "+-->");
|
|
}
|
|
|
|
static int amdfw_bios_dir_walk(FILE *fw, uint32_t bios_offset, uint32_t cookie, uint8_t level)
|
|
{
|
|
bios_directory_entry *current_entries = NULL;
|
|
size_t num_current_entries = 0;
|
|
bios_directory_hdr header;
|
|
uint32_t l2_dir_offset = 0;
|
|
char indent[MAX_INDENTATION_LEN] = {0};
|
|
|
|
if (read_bios_directory(fw, bios_offset, cookie, &header,
|
|
¤t_entries, &num_current_entries) != 0)
|
|
return 1;
|
|
|
|
do_indentation_string(indent, level);
|
|
for (size_t i = 0; i < num_current_entries; i++) {
|
|
uint32_t type = current_entries[i].type;
|
|
uint64_t mode = current_entries[i].address_mode;
|
|
uint64_t addr = current_entries[i].source;
|
|
|
|
if (type == AMD_BIOS_APOB || type == AMD_BIOS_PSP_SHARED_MEM)
|
|
printf("%sBIOS%s: 0x%02x 0x%lx(DRAM-Address)\n",
|
|
indent, cookie == BHD_COOKIE ? "L1" : "L2",
|
|
type, current_entries[i].dest);
|
|
else if (type == AMD_BIOS_APOB_NV)
|
|
printf("%sBIOS%s: 0x%02x 0x%08lx 0x%08x\n",
|
|
indent, cookie == BHD_COOKIE ? "L1" : "L2",
|
|
type, relative_offset(bios_offset, addr, AMD_ADDR_PHYSICAL),
|
|
current_entries[i].size);
|
|
else
|
|
printf("%sBIOS%s: 0x%02x 0x%08lx 0x%08x\n",
|
|
indent, cookie == BHD_COOKIE ? "L1" : "L2",
|
|
type, relative_offset(bios_offset, addr, mode),
|
|
current_entries[i].size);
|
|
|
|
if (type == AMD_BIOS_L2_PTR) {
|
|
/* There's a second level BIOS directory to read */
|
|
if (l2_dir_offset != 0) {
|
|
ERR("Duplicate BIOS L2 Entry, prior offset: %08x\n",
|
|
l2_dir_offset);
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
l2_dir_offset = relative_offset(bios_offset, addr, mode);
|
|
printf(" %sBIOSL2: Dir 0x%08x\n", indent, l2_dir_offset);
|
|
amdfw_bios_dir_walk(fw, l2_dir_offset, BHDL2_COOKIE, level + 2);
|
|
}
|
|
}
|
|
|
|
free(current_entries);
|
|
return 0;
|
|
}
|
|
|
|
static int amdfw_psp_dir_walk(FILE *fw, uint32_t psp_offset, uint32_t cookie, uint8_t level)
|
|
{
|
|
psp_directory_entry *current_entries = NULL;
|
|
size_t num_current_entries = 0;
|
|
psp_directory_header header;
|
|
uint32_t l2_dir_offset = 0;
|
|
uint32_t bios_dir_offset = 0;
|
|
uint32_t ish_dir_offset = 0;
|
|
ish_directory_table ish_dir;
|
|
char indent[MAX_INDENTATION_LEN] = {0};
|
|
|
|
if (read_psp_directory(fw, psp_offset, cookie, &header,
|
|
¤t_entries, &num_current_entries) != 0)
|
|
return 1;
|
|
|
|
do_indentation_string(indent, level);
|
|
for (size_t i = 0; i < num_current_entries; i++) {
|
|
uint32_t type = current_entries[i].type;
|
|
uint64_t mode = current_entries[i].address_mode;
|
|
uint64_t addr = current_entries[i].addr;
|
|
|
|
if (type == AMD_PSP_FUSE_CHAIN)
|
|
printf("%sPSP%s: 0x%02x 0x%lx(Soft-fuse)\n",
|
|
indent, cookie == PSP_COOKIE ? "L1" : "L2",
|
|
type, mode << 62 | addr);
|
|
else
|
|
printf("%sPSP%s: 0x%02x 0x%08lx 0x%08x\n",
|
|
indent, cookie == PSP_COOKIE ? "L1" : "L2",
|
|
type, relative_offset(psp_offset, addr, mode),
|
|
current_entries[i].size);
|
|
|
|
switch (type) {
|
|
case AMD_FW_L2_PTR:
|
|
/* There's a second level PSP directory to read */
|
|
if (l2_dir_offset != 0) {
|
|
ERR("Duplicate PSP L2 Entry, prior offset: %08x\n",
|
|
l2_dir_offset);
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
l2_dir_offset = relative_offset(psp_offset, addr, mode);
|
|
printf(" %sPSPL2: Dir 0x%08x\n", indent, l2_dir_offset);
|
|
amdfw_psp_dir_walk(fw, l2_dir_offset, PSPL2_COOKIE, level + 2);
|
|
break;
|
|
|
|
case AMD_FW_RECOVERYAB_A:
|
|
if (l2_dir_offset != 0) {
|
|
ERR("Duplicate PSP L2 Entry, prior offset: %08x\n",
|
|
l2_dir_offset);
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
ish_dir_offset = relative_offset(psp_offset, addr, mode);
|
|
if (read_ish_directory(fw, ish_dir_offset, &ish_dir) != 0) {
|
|
ERR("Error reading ISH directory\n");
|
|
free(current_entries);
|
|
return 1;
|
|
}
|
|
|
|
l2_dir_offset = ish_dir.pl2_location;
|
|
printf(" %sPSPL2: Dir 0x%08x\n", indent, l2_dir_offset);
|
|
amdfw_psp_dir_walk(fw, l2_dir_offset, PSPL2_COOKIE, level + 2);
|
|
break;
|
|
|
|
case AMD_FW_BIOS_TABLE:
|
|
bios_dir_offset = relative_offset(psp_offset, addr, mode);
|
|
printf(" %sBIOSL2: Dir 0x%08x\n", indent, bios_dir_offset);
|
|
amdfw_bios_dir_walk(fw, bios_dir_offset, BHDL2_COOKIE, level + 2);
|
|
break;
|
|
|
|
default:
|
|
/* No additional processing required, continue to the next entry. */
|
|
break;
|
|
}
|
|
}
|
|
|
|
free(current_entries);
|
|
return 0;
|
|
}
|
|
|
|
static int list_amdfw_psp_dir(FILE *fw, const embedded_firmware *fw_header)
|
|
{
|
|
uint32_t psp_offset = 0;
|
|
|
|
/* 0xffffffff indicates that the offset is in new_psp_directory */
|
|
if (fw_header->psp_directory != 0xffffffff)
|
|
psp_offset = fw_header->psp_directory;
|
|
else
|
|
psp_offset = fw_header->new_psp_directory;
|
|
|
|
printf("PSPL1: Dir 0x%08x\n", psp_offset);
|
|
amdfw_psp_dir_walk(fw, psp_offset, PSP_COOKIE, 0);
|
|
return 0;
|
|
}
|
|
|
|
static int list_amdfw_bios_dir(FILE *fw, const embedded_firmware *fw_header)
|
|
{
|
|
/* 0xffffffff implies that the SoC uses recovery A/B layout. Only BIOS L2 directory
|
|
is present and that too as part of PSP L2 directory. */
|
|
if (fw_header->bios3_entry != 0xffffffff) {
|
|
printf("BIOSL1: Dir 0x%08x\n", fw_header->bios3_entry);
|
|
amdfw_bios_dir_walk(fw, fw_header->bios3_entry, BHD_COOKIE, 0);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int list_amdfw_ro(FILE *fw, const embedded_firmware *fw_header)
|
|
{
|
|
printf("Table: FW Offset Size\n");
|
|
list_amdfw_psp_dir(fw, fw_header);
|
|
list_amdfw_bios_dir(fw, fw_header);
|
|
return 0;
|
|
}
|
|
|
|
enum {
|
|
AMDFW_OPT_HELP = 'h',
|
|
AMDFW_OPT_SOFT_FUSE = 1UL << 0, /* Print Softfuse */
|
|
AMDFW_OPT_RO_LIST = 1UL << 1, /* List entries in AMDFW RO */
|
|
};
|
|
|
|
static char const optstring[] = {AMDFW_OPT_HELP};
|
|
|
|
static struct option long_options[] = {
|
|
{"help", no_argument, 0, AMDFW_OPT_HELP},
|
|
{"soft-fuse", no_argument, 0, AMDFW_OPT_SOFT_FUSE},
|
|
{"ro-list", no_argument, 0, AMDFW_OPT_RO_LIST},
|
|
};
|
|
|
|
static void print_usage(void)
|
|
{
|
|
printf("amdfwread: Examine AMD firmware images\n");
|
|
printf("Usage: amdfwread [options] <file>\n");
|
|
printf("--soft-fuse Print soft fuse value\n");
|
|
printf("--ro-list List the programs under AMDFW in RO region\n");
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
char *fw_file = NULL;
|
|
|
|
int selected_functions = 0;
|
|
while (1) {
|
|
int opt = getopt_long(argc, argv, optstring, long_options, NULL);
|
|
|
|
if (opt == -1) {
|
|
if (optind != (argc - 1)) {
|
|
/* Print usage if one and only one option i.e. filename is
|
|
not found. */
|
|
print_usage();
|
|
return 0;
|
|
}
|
|
|
|
fw_file = argv[optind];
|
|
break;
|
|
}
|
|
|
|
switch (opt) {
|
|
case AMDFW_OPT_HELP:
|
|
print_usage();
|
|
return 0;
|
|
|
|
case AMDFW_OPT_SOFT_FUSE:
|
|
case AMDFW_OPT_RO_LIST:
|
|
selected_functions |= opt;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
FILE *fw = fopen(fw_file, "rb");
|
|
if (!fw) {
|
|
ERR("Failed to open FW file %s\n", fw_file);
|
|
return 1;
|
|
}
|
|
|
|
/* Find the FW header by checking each possible location */
|
|
embedded_firmware fw_header;
|
|
int found_header = 0;
|
|
for (size_t i = 0; i < ARRAY_SIZE(fw_header_offsets); i++) {
|
|
if (read_fw_header(fw, fw_header_offsets[i], &fw_header) == 0) {
|
|
found_header = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found_header) {
|
|
ERR("Failed to find FW header\n");
|
|
fclose(fw);
|
|
return 1;
|
|
}
|
|
|
|
if (selected_functions & AMDFW_OPT_SOFT_FUSE) {
|
|
if (read_soft_fuse(fw, &fw_header) != 0) {
|
|
fclose(fw);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (selected_functions & AMDFW_OPT_RO_LIST) {
|
|
if (list_amdfw_ro(fw, &fw_header) != 0) {
|
|
fclose(fw);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
fclose(fw);
|
|
return 0;
|
|
}
|