-
Notifications
You must be signed in to change notification settings - Fork 2
/
archive.c
115 lines (101 loc) · 2.92 KB
/
archive.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
/**
* @file archive.c
* A collection of files
* @author Marko Mäkelä (marko.makela at iki.fi)
*/
/*
** Copyright © 1998, 2022, Marko Mäkelä
**
** This program is free software; you can redistribute it and/or modify
** it under the terms of the GNU General Public License as published by
** the Free Software Foundation; either version 2 of the License, or
** (at your option) any later version.
**
** This program is distributed in the hope that it will be useful,
** but WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
** GNU General Public License for more details.
**
** You should have received a copy of the GNU General Public License
** along with this program; if not, write to the Free Software
** Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include "output.h"
/** Allocate an archive data structure.
* @return a newly allocated empty archive structure
*/
struct Archive*
newArchive (void)
{
return calloc (1, sizeof (struct Archive));
}
/** Deallocate an archive data structure.
* @param archive the archive to be deallocated
*/
void deleteArchive (struct Archive* archive)
{
while (archive->first) {
struct ArchiveEntry* ae = archive->first->next;
free (archive->first->data);
free (archive->first);
archive->first = ae;
}
free (archive);
}
/** Write a file to an archive.
* @param name native (PETSCII) name of the file
* @param data the contents of the file
* @param length length of the file contents
* @param archive the archive the file is written to
* @param log Call-back function for diagnostic output
* @return status of the operation
*/
enum WrStatus
WriteArchive (const struct Filename* name,
const byte_t* data,
size_t length,
struct Archive* archive,
log_t log)
{
struct ArchiveEntry* ae;
switch (name->type) {
default:
(*log) (Errors, name, "Unsupported file type.");
return WrFail;
case DEL:
case SEQ:
case PRG:
case USR:
case REL:
break;
}
/* check for duplicate file names */
if (!allowDuplicates)
for (ae = archive->first; ae; ae = ae->next)
if (!memcmp (&ae->name.name, name->name, sizeof name->name))
return WrFileExists;
if (!(ae = malloc (sizeof (*ae)))) {
(*log) (Errors, name, "Out of memory.");
return WrNoSpace;
}
if (!(ae->data = malloc (length))) {
free (ae);
(*log) (Errors, name, "Out of memory.");
return WrNoSpace;
}
memcpy (&ae->name, name, sizeof (*name));
if (length)
memcpy (ae->data, data, length);
ae->length = length;
ae->next = 0;
if (archive->last)
archive->last->next = ae;
else
archive->first = ae;
archive->last = ae;
return WrOK;
}