sys/acpi: Bring in Intel's acpica-unix2-20131218 verbatim.
[dragonfly.git] / sys / contrib / dev / acpica / source / tools / acpisrc / asfile.c
1 /******************************************************************************
2  *
3  * Module Name: asfile - Main module for the acpi source processor utility
4  *
5  *****************************************************************************/
6
7 /*
8  * Copyright (C) 2000 - 2013, 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 "acpisrc.h"
45
46 /* Local prototypes */
47
48 void
49 AsDoWildcard (
50     ACPI_CONVERSION_TABLE   *ConversionTable,
51     char                    *SourcePath,
52     char                    *TargetPath,
53     int                     MaxPathLength,
54     int                     FileType,
55     char                    *WildcardSpec);
56
57 BOOLEAN
58 AsDetectLoneLineFeeds (
59     char                    *Filename,
60     char                    *Buffer);
61
62 static ACPI_INLINE int
63 AsMaxInt (int a, int b)
64 {
65     return (a > b ? a : b);
66 }
67
68
69 /******************************************************************************
70  *
71  * FUNCTION:    AsDoWildcard
72  *
73  * DESCRIPTION: Process files via wildcards
74  *
75  ******************************************************************************/
76
77 void
78 AsDoWildcard (
79     ACPI_CONVERSION_TABLE   *ConversionTable,
80     char                    *SourcePath,
81     char                    *TargetPath,
82     int                     MaxPathLength,
83     int                     FileType,
84     char                    *WildcardSpec)
85 {
86     void                    *DirInfo;
87     char                    *Filename;
88     char                    *SourceDirPath;
89     char                    *TargetDirPath;
90     char                    RequestedFileType;
91
92
93     if (FileType == FILE_TYPE_DIRECTORY)
94     {
95         RequestedFileType = REQUEST_DIR_ONLY;
96     }
97     else
98     {
99         RequestedFileType = REQUEST_FILE_ONLY;
100     }
101
102     VERBOSE_PRINT (("Checking for %s source files in directory \"%s\"\n",
103             WildcardSpec, SourcePath));
104
105     /* Open the directory for wildcard search */
106
107     DirInfo = AcpiOsOpenDirectory (SourcePath, WildcardSpec, RequestedFileType);
108     if (DirInfo)
109     {
110         /*
111          * Get all of the files that match both the
112          * wildcard and the requested file type
113          */
114         while ((Filename = AcpiOsGetNextFilename (DirInfo)))
115         {
116             /* Looking for directory files, must check file type */
117
118             switch (RequestedFileType)
119             {
120             case REQUEST_DIR_ONLY:
121
122                 /* If we actually have a dir, process the subtree */
123
124                 if (!AsCheckForDirectory (SourcePath, TargetPath, Filename,
125                         &SourceDirPath, &TargetDirPath))
126                 {
127                     VERBOSE_PRINT (("Subdirectory: %s\n", Filename));
128
129                     AsProcessTree (ConversionTable, SourceDirPath, TargetDirPath);
130                     free (SourceDirPath);
131                     free (TargetDirPath);
132                 }
133                 break;
134
135             case REQUEST_FILE_ONLY:
136
137                 /* Otherwise, this is a file, not a directory */
138
139                 VERBOSE_PRINT (("File: %s\n", Filename));
140
141                 AsProcessOneFile (ConversionTable, SourcePath, TargetPath,
142                         MaxPathLength, Filename, FileType);
143                 break;
144
145             default:
146
147                 break;
148             }
149         }
150
151         /* Cleanup */
152
153         AcpiOsCloseDirectory (DirInfo);
154     }
155 }
156
157
158 /******************************************************************************
159  *
160  * FUNCTION:    AsProcessTree
161  *
162  * DESCRIPTION: Process the directory tree. Files with the extension ".C" and
163  *              ".H" are processed as the tree is traversed.
164  *
165  ******************************************************************************/
166
167 ACPI_NATIVE_INT
168 AsProcessTree (
169     ACPI_CONVERSION_TABLE   *ConversionTable,
170     char                    *SourcePath,
171     char                    *TargetPath)
172 {
173     int                     MaxPathLength;
174
175
176     MaxPathLength = AsMaxInt (strlen (SourcePath), strlen (TargetPath));
177
178     if (!(ConversionTable->Flags & FLG_NO_FILE_OUTPUT))
179     {
180         if (ConversionTable->Flags & FLG_LOWERCASE_DIRNAMES)
181         {
182             AsStrlwr (TargetPath);
183         }
184
185         VERBOSE_PRINT (("Creating Directory \"%s\"\n", TargetPath));
186         if (mkdir (TargetPath))
187         {
188             if (errno != EEXIST)
189             {
190                 printf ("Could not create target directory\n");
191                 return (-1);
192             }
193         }
194     }
195
196     /* Do the C source files */
197
198     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
199             FILE_TYPE_SOURCE, "*.c");
200
201     /* Do the C header files */
202
203     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
204             FILE_TYPE_HEADER, "*.h");
205
206     /* Do the Lex file(s) */
207
208     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
209             FILE_TYPE_SOURCE, "*.l");
210
211     /* Do the yacc file(s) */
212
213     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
214             FILE_TYPE_SOURCE, "*.y");
215
216     /* Do any ASL files */
217
218     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
219             FILE_TYPE_HEADER, "*.asl");
220
221     /* Do any subdirectories */
222
223     AsDoWildcard (ConversionTable, SourcePath, TargetPath, MaxPathLength,
224             FILE_TYPE_DIRECTORY, "*");
225
226     return (0);
227 }
228
229
230 /******************************************************************************
231  *
232  * FUNCTION:    AsDetectLoneLineFeeds
233  *
234  * DESCRIPTION: Find LF without CR.
235  *
236  ******************************************************************************/
237
238 BOOLEAN
239 AsDetectLoneLineFeeds (
240     char                    *Filename,
241     char                    *Buffer)
242 {
243     UINT32                  i = 1;
244     UINT32                  LfCount = 0;
245     UINT32                  LineCount = 0;
246
247
248     if (!Buffer[0])
249     {
250         return (FALSE);
251     }
252
253     while (Buffer[i])
254     {
255         if (Buffer[i] == 0x0A)
256         {
257             if (Buffer[i-1] != 0x0D)
258             {
259                 LfCount++;
260             }
261             LineCount++;
262         }
263         i++;
264     }
265
266     if (LfCount)
267     {
268         if (LineCount == LfCount)
269         {
270             if (!Gbl_IgnoreLoneLineFeeds)
271             {
272                 printf ("%s: ****File has UNIX format**** (LF only, not CR/LF) %u lines\n",
273                     Filename, LfCount);
274             }
275         }
276         else
277         {
278             printf ("%s: %u lone linefeeds in file\n", Filename, LfCount);
279         }
280         return (TRUE);
281     }
282
283     return (FALSE);
284 }
285
286
287 /******************************************************************************
288  *
289  * FUNCTION:    AsConvertFile
290  *
291  * DESCRIPTION: Perform the requested transforms on the file buffer (as
292  *              determined by the ConversionTable and the FileType).
293  *
294  ******************************************************************************/
295
296 void
297 AsConvertFile (
298     ACPI_CONVERSION_TABLE   *ConversionTable,
299     char                    *FileBuffer,
300     char                    *Filename,
301     ACPI_NATIVE_INT         FileType)
302 {
303     UINT32                  i;
304     UINT32                  Functions;
305     ACPI_STRING_TABLE       *StringTable;
306     ACPI_IDENTIFIER_TABLE   *ConditionalTable;
307     ACPI_IDENTIFIER_TABLE   *LineTable;
308     ACPI_IDENTIFIER_TABLE   *MacroTable;
309     ACPI_TYPED_IDENTIFIER_TABLE *StructTable;
310     ACPI_IDENTIFIER_TABLE   *SpecialMacroTable;
311
312
313     switch (FileType)
314     {
315     case FILE_TYPE_SOURCE:
316
317         Functions           = ConversionTable->SourceFunctions;
318         StringTable         = ConversionTable->SourceStringTable;
319         LineTable           = ConversionTable->SourceLineTable;
320         ConditionalTable    = ConversionTable->SourceConditionalTable;
321         MacroTable          = ConversionTable->SourceMacroTable;
322         StructTable         = ConversionTable->SourceStructTable;
323         SpecialMacroTable   = ConversionTable->SourceSpecialMacroTable;
324        break;
325
326     case FILE_TYPE_HEADER:
327
328         Functions           = ConversionTable->HeaderFunctions;
329         StringTable         = ConversionTable->HeaderStringTable;
330         LineTable           = ConversionTable->HeaderLineTable;
331         ConditionalTable    = ConversionTable->HeaderConditionalTable;
332         MacroTable          = ConversionTable->HeaderMacroTable;
333         StructTable         = ConversionTable->HeaderStructTable;
334         SpecialMacroTable   = ConversionTable->HeaderSpecialMacroTable;
335         break;
336
337     default:
338
339         printf ("Unknown file type, cannot process\n");
340         return;
341     }
342
343
344     Gbl_StructDefs = strstr (FileBuffer, "/* acpisrc:StructDefs");
345     Gbl_Files++;
346     VERBOSE_PRINT (("Processing %u bytes\n",
347         (unsigned int) strlen (FileBuffer)));
348
349     if (Gbl_Cleanup)
350     {
351         AsRemoveExtraLines (FileBuffer, Filename);
352         AsRemoveSpacesAfterPeriod (FileBuffer, Filename);
353     }
354
355     if (ConversionTable->LowerCaseTable)
356     {
357         for (i = 0; ConversionTable->LowerCaseTable[i].Identifier; i++)
358         {
359             AsLowerCaseString (ConversionTable->LowerCaseTable[i].Identifier,
360                                 FileBuffer);
361         }
362     }
363
364     /* Process all the string replacements */
365
366     if (StringTable)
367     {
368         for (i = 0; StringTable[i].Target; i++)
369         {
370             AsReplaceString (StringTable[i].Target, StringTable[i].Replacement,
371                     StringTable[i].Type, FileBuffer);
372         }
373     }
374
375     if (LineTable)
376     {
377         for (i = 0; LineTable[i].Identifier; i++)
378         {
379             AsRemoveLine (FileBuffer, LineTable[i].Identifier);
380         }
381     }
382
383     if (ConditionalTable)
384     {
385         for (i = 0; ConditionalTable[i].Identifier; i++)
386         {
387             AsRemoveConditionalCompile (FileBuffer, ConditionalTable[i].Identifier);
388         }
389     }
390
391     if (MacroTable)
392     {
393         for (i = 0; MacroTable[i].Identifier; i++)
394         {
395             AsRemoveMacro (FileBuffer, MacroTable[i].Identifier);
396         }
397     }
398
399     if (StructTable)
400     {
401         for (i = 0; StructTable[i].Identifier; i++)
402         {
403             AsInsertPrefix (FileBuffer, StructTable[i].Identifier, StructTable[i].Type);
404         }
405     }
406
407     if (SpecialMacroTable)
408     {
409         for (i = 0; SpecialMacroTable[i].Identifier; i++)
410         {
411             AsCleanupSpecialMacro (FileBuffer, SpecialMacroTable[i].Identifier);
412         }
413     }
414
415     /* Process the function table */
416
417     for (i = 0; i < 32; i++)
418     {
419         /* Decode the function bitmap */
420
421         switch ((1 << i) & Functions)
422         {
423         case 0:
424
425             /* This function not configured */
426             break;
427
428         case CVT_COUNT_TABS:
429
430             AsCountTabs (FileBuffer, Filename);
431             break;
432
433         case CVT_COUNT_NON_ANSI_COMMENTS:
434
435             AsCountNonAnsiComments (FileBuffer, Filename);
436             break;
437
438         case CVT_CHECK_BRACES:
439
440             AsCheckForBraces (FileBuffer, Filename);
441             break;
442
443         case CVT_TRIM_LINES:
444
445             AsTrimLines (FileBuffer, Filename);
446             break;
447
448         case CVT_COUNT_LINES:
449
450             AsCountSourceLines (FileBuffer, Filename);
451             break;
452
453         case CVT_BRACES_ON_SAME_LINE:
454
455             AsBracesOnSameLine (FileBuffer);
456             break;
457
458         case CVT_MIXED_CASE_TO_UNDERSCORES:
459
460             AsMixedCaseToUnderscores (FileBuffer, Filename);
461             break;
462
463         case CVT_LOWER_CASE_IDENTIFIERS:
464
465             AsLowerCaseIdentifiers (FileBuffer);
466             break;
467
468         case CVT_REMOVE_DEBUG_MACROS:
469
470             AsRemoveDebugMacros (FileBuffer);
471             break;
472
473         case CVT_TRIM_WHITESPACE:
474
475             AsTrimWhitespace (FileBuffer);
476             break;
477
478         case CVT_REMOVE_EMPTY_BLOCKS:
479
480             AsRemoveEmptyBlocks (FileBuffer, Filename);
481             break;
482
483         case CVT_REDUCE_TYPEDEFS:
484
485             AsReduceTypedefs (FileBuffer, "typedef union");
486             AsReduceTypedefs (FileBuffer, "typedef struct");
487             break;
488
489         case CVT_SPACES_TO_TABS4:
490
491             AsTabify4 (FileBuffer);
492             break;
493
494         case CVT_SPACES_TO_TABS8:
495
496             AsTabify8 (FileBuffer);
497             break;
498
499         case CVT_COUNT_SHORTMULTILINE_COMMENTS:
500
501 #ifdef ACPI_FUTURE_IMPLEMENTATION
502             AsTrimComments (FileBuffer, Filename);
503 #endif
504             break;
505
506         default:
507
508             printf ("Unknown conversion subfunction opcode\n");
509             break;
510         }
511     }
512
513     if (ConversionTable->NewHeader)
514     {
515         AsReplaceHeader (FileBuffer, ConversionTable->NewHeader);
516     }
517 }
518
519
520 /******************************************************************************
521  *
522  * FUNCTION:    AsProcessOneFile
523  *
524  * DESCRIPTION: Process one source file. The file is opened, read entirely
525  *              into a buffer, converted, then written to a new file.
526  *
527  ******************************************************************************/
528
529 ACPI_NATIVE_INT
530 AsProcessOneFile (
531     ACPI_CONVERSION_TABLE   *ConversionTable,
532     char                    *SourcePath,
533     char                    *TargetPath,
534     int                     MaxPathLength,
535     char                    *Filename,
536     ACPI_NATIVE_INT         FileType)
537 {
538     char                    *Pathname;
539     char                    *OutPathname = NULL;
540
541
542     /* Allocate a file pathname buffer for both source and target */
543
544     Pathname = calloc (MaxPathLength + strlen (Filename) + 2, 1);
545     if (!Pathname)
546     {
547         printf ("Could not allocate buffer for file pathnames\n");
548         return (-1);
549     }
550
551     Gbl_FileType = FileType;
552
553     /* Generate the source pathname and read the file */
554
555     if (SourcePath)
556     {
557         strcpy (Pathname, SourcePath);
558         strcat (Pathname, "/");
559     }
560
561     strcat (Pathname, Filename);
562
563     if (AsGetFile (Pathname, &Gbl_FileBuffer, &Gbl_FileSize))
564     {
565         return (-1);
566     }
567
568     Gbl_HeaderSize = 0;
569     if (strstr (Filename, ".asl"))
570     {
571         Gbl_HeaderSize = LINES_IN_ASL_HEADER; /* Lines in default ASL header */
572     }
573     else if (strstr (Gbl_FileBuffer, LEGAL_HEADER_SIGNATURE))
574     {
575         Gbl_HeaderSize = LINES_IN_LEGAL_HEADER; /* Normal C file and H header */
576     }
577     else if (strstr (Gbl_FileBuffer, LINUX_HEADER_SIGNATURE))
578     {
579         Gbl_HeaderSize = LINES_IN_LINUX_HEADER; /* Linuxized C file and H header */
580     }
581
582     /* Process the file in the buffer */
583
584     Gbl_MadeChanges = FALSE;
585     if (!Gbl_IgnoreLoneLineFeeds && Gbl_HasLoneLineFeeds)
586     {
587         /*
588          * All lone LFs will be converted to CR/LF
589          * (when file is written, Windows version only)
590          */
591         printf ("Converting lone linefeeds\n");
592         Gbl_MadeChanges = TRUE;
593     }
594
595     AsConvertFile (ConversionTable, Gbl_FileBuffer, Pathname, FileType);
596
597     if (!(ConversionTable->Flags & FLG_NO_FILE_OUTPUT))
598     {
599         if (!(Gbl_Overwrite && !Gbl_MadeChanges))
600         {
601             /* Generate the target pathname and write the file */
602
603             OutPathname = calloc (MaxPathLength + strlen (Filename) + 2 + strlen (TargetPath), 1);
604             if (!OutPathname)
605             {
606                 printf ("Could not allocate buffer for file pathnames\n");
607                 return (-1);
608             }
609
610             strcpy (OutPathname, TargetPath);
611             if (SourcePath)
612             {
613                 strcat (OutPathname, "/");
614                 strcat (OutPathname, Filename);
615             }
616
617             AsPutFile (OutPathname, Gbl_FileBuffer, ConversionTable->Flags);
618         }
619     }
620
621     free (Gbl_FileBuffer);
622     free (Pathname);
623     if (OutPathname)
624     {
625         free (OutPathname);
626     }
627
628     return (0);
629 }
630
631
632 /******************************************************************************
633  *
634  * FUNCTION:    AsCheckForDirectory
635  *
636  * DESCRIPTION: Check if the current file is a valid directory. If not,
637  *              construct the full pathname for the source and target paths.
638  *              Checks for the dot and dot-dot files (they are ignored)
639  *
640  ******************************************************************************/
641
642 ACPI_NATIVE_INT
643 AsCheckForDirectory (
644     char                    *SourceDirPath,
645     char                    *TargetDirPath,
646     char                    *Filename,
647     char                    **SourcePath,
648     char                    **TargetPath)
649 {
650     char                    *SrcPath;
651     char                    *TgtPath;
652
653
654     if (!(strcmp (Filename, ".")) ||
655         !(strcmp (Filename, "..")))
656     {
657         return (-1);
658     }
659
660     SrcPath = calloc (strlen (SourceDirPath) + strlen (Filename) + 2, 1);
661     if (!SrcPath)
662     {
663         printf ("Could not allocate buffer for directory source pathname\n");
664         return (-1);
665     }
666
667     TgtPath = calloc (strlen (TargetDirPath) + strlen (Filename) + 2, 1);
668     if (!TgtPath)
669     {
670         printf ("Could not allocate buffer for directory target pathname\n");
671         free (SrcPath);
672         return (-1);
673     }
674
675     strcpy (SrcPath, SourceDirPath);
676     strcat (SrcPath, "/");
677     strcat (SrcPath, Filename);
678
679     strcpy (TgtPath, TargetDirPath);
680     strcat (TgtPath, "/");
681     strcat (TgtPath, Filename);
682
683     *SourcePath = SrcPath;
684     *TargetPath = TgtPath;
685     return (0);
686 }
687
688
689 /******************************************************************************
690  *
691  * FUNCTION:    AsGetFile
692  *
693  * DESCRIPTION: Open a file and read it entirely into a an allocated buffer
694  *
695  ******************************************************************************/
696
697 int
698 AsGetFile (
699     char                    *Filename,
700     char                    **FileBuffer,
701     UINT32                  *FileSize)
702 {
703     FILE                    *File;
704     UINT32                  Size;
705     char                    *Buffer;
706     int                     Seek1;
707     int                     Seek2;
708     size_t                  Actual;
709
710
711     /* Binary mode leaves CR/LF pairs */
712
713     File = fopen (Filename, "rb");
714     if (!File)
715     {
716         printf ("Could not open file %s\n", Filename);
717         return (-1);
718     }
719
720     /* Need file size to allocate a buffer */
721
722     Seek1 = fseek (File, 0L, SEEK_END);
723     Size = ftell (File);
724     Seek2 = fseek (File, 0L, SEEK_SET);
725
726     if (Seek1 || Seek2 || (Size == -1))
727     {
728         printf ("Could not get file size for %s\n", Filename);
729         goto ErrorExit;
730     }
731
732     /*
733      * Create a buffer for the entire file
734      * Add plenty extra buffer to accommodate string replacements
735      */
736     Gbl_TotalSize += Size;
737
738     Buffer = calloc (Size * 2, 1);
739     if (!Buffer)
740     {
741         printf ("Could not allocate buffer of size %u\n", Size * 2);
742         goto ErrorExit;
743     }
744
745     /* Read the entire file */
746
747     Actual = fread (Buffer, 1, Size, File);
748     if (Actual != Size)
749     {
750         printf ("Could not read the input file %s (%u bytes)\n",
751             Filename, Size);
752         goto ErrorExit;
753     }
754
755     Buffer [Size] = 0;         /* Null terminate the buffer */
756     fclose (File);
757
758     /* Check for unix contamination */
759
760     Gbl_HasLoneLineFeeds = AsDetectLoneLineFeeds (Filename, Buffer);
761
762     /*
763      * Convert all CR/LF pairs to LF only. We do this locally so that
764      * this code is portable across operating systems.
765      */
766     AsConvertToLineFeeds (Buffer);
767
768     *FileBuffer = Buffer;
769     *FileSize = Size;
770     return (0);
771
772
773 ErrorExit:
774
775     fclose (File);
776     return (-1);
777 }
778
779
780 /******************************************************************************
781  *
782  * FUNCTION:    AsPutFile
783  *
784  * DESCRIPTION: Create a new output file and write the entire contents of the
785  *              buffer to the new file. Buffer must be a zero terminated string
786  *
787  ******************************************************************************/
788
789 int
790 AsPutFile (
791     char                    *Pathname,
792     char                    *FileBuffer,
793     UINT32                  SystemFlags)
794 {
795     FILE                    *File;
796     UINT32                  FileSize;
797     size_t                  Actual;
798     int                     Status = 0;
799
800
801     /* Create the target file */
802
803     if (!(SystemFlags & FLG_NO_CARRIAGE_RETURNS))
804     {
805         /* Put back the CR before each LF */
806
807         AsInsertCarriageReturns (FileBuffer);
808     }
809
810     File = fopen (Pathname, "w+b");
811     if (!File)
812     {
813         perror ("Could not create destination file");
814         printf ("Could not create destination file \"%s\"\n", Pathname);
815         return (-1);
816     }
817
818     /* Write the buffer to the file */
819
820     FileSize = strlen (FileBuffer);
821     Actual = fwrite (FileBuffer, 1, FileSize, File);
822     if (Actual != FileSize)
823     {
824         printf ("Error writing output file \"%s\"\n", Pathname);
825         Status = -1;
826     }
827
828     fclose (File);
829     return (Status);
830 }