-
Notifications
You must be signed in to change notification settings - Fork 0
/
cksum.c
232 lines (195 loc) · 5.76 KB
/
cksum.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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
/*
* Copyright (c) 2003 Regents of The University of Michigan.
* All Rights Reserved. See COPYRIGHT.
*/
#include "config.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/param.h>
#ifdef __APPLE__
#include <sys/paths.h>
#endif /* __APPLE__ */
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <openssl/evp.h>
#include "openssl_compat.h" // Compatibility shims for OpenSSL < 1.1.0
#include "applefile.h"
#include "cksum.h"
#include "base64.h"
#ifdef ENABLE_XATTR
#include "xattr.h"
#endif /* ENABLE_XATTR */
/*
* do_cksum calculates the checksum for PATH and returns it base64 encoded
* in cksum_b64 which must be of size SZ_BASE64_E( EVP_MAX_MD_SIZE ).
*
* return values:
* < 0 system error: errno set, no message given
* >= 0 number of bytes check summed
*/
off_t
do_fcksum( int fd, char *cksum_b64 )
{
unsigned int md_len;
ssize_t rr;
off_t size = 0;
unsigned char buf[ 8192 ];
extern EVP_MD *md;
EVP_MD_CTX *mdctx = EVP_MD_CTX_new();
unsigned char md_value[ SZ_BASE64_D( SZ_BASE64_E( EVP_MAX_MD_SIZE ) ) ];
EVP_DigestInit( mdctx, md );
while (( rr = read( fd, buf, sizeof( buf ))) > 0 ) {
size += rr;
EVP_DigestUpdate( mdctx, buf, (unsigned int)rr );
}
if ( rr < 0 ) {
return( -1 );
}
EVP_DigestFinal( mdctx, md_value, &md_len );
base64_e( md_value, md_len, cksum_b64 );
EVP_MD_CTX_free( mdctx );
return( size );
}
off_t
do_cksum( char *path, char *cksum_b64 )
{
int fd;
off_t size = 0;
if (( fd = open( path, O_RDONLY, 0 )) < 0 ) {
return( -1 );
}
size = do_fcksum( fd, cksum_b64 );
if ( close( fd ) != 0 ) {
return( -1 );
}
return( size );
}
#ifdef __APPLE__
/*
* do_acksum calculates the checksum for the encoded apple single file of PATH
* and returns it base64 encoded in cksum_b64 which must be of size
* SZ_BASE64_E( EVP_MAX_MD_SIZE ).
*
* return values:
* >= 0 number of bytes check summed
* < 0 system error: errno set, no message given
*
* do_acksum should only be called on native HFS+ system.
*/
off_t
do_acksum( char *path, char *cksum_b64, struct applefileinfo *afinfo )
{
int dfd, rfd, rc;
char buf[ 8192 ], rsrc_path[ MAXPATHLEN ];
off_t size = 0;
extern struct as_header as_header;
struct as_entry as_entries_endian[ 3 ];
unsigned int md_len;
extern EVP_MD *md;
EVP_MD_CTX *mdctx = EVP_MD_CTX_new();
unsigned char md_value[ SZ_BASE64_D( SZ_BASE64_E( EVP_MAX_MD_SIZE ) ) ];
EVP_DigestInit( mdctx, md );
/* checksum applesingle header */
EVP_DigestUpdate( mdctx, (char *)&as_header, AS_HEADERLEN );
size += (size_t)AS_HEADERLEN;
/* endian handling, sum big-endian header entries */
memcpy( &as_entries_endian, &afinfo->as_ents,
( 3 * sizeof( struct as_entry )));
as_entry_netswap( &as_entries_endian[ AS_FIE ] );
as_entry_netswap( &as_entries_endian[ AS_RFE ] );
as_entry_netswap( &as_entries_endian[ AS_DFE ] );
/* checksum header entries */
EVP_DigestUpdate( mdctx, (char *)&as_entries_endian,
(unsigned int)( 3 * sizeof( struct as_entry )));
size += sizeof( 3 * sizeof( struct as_entry ));
/* checksum finder info data */
EVP_DigestUpdate( mdctx, afinfo->ai.ai_data, FINFOLEN );
size += FINFOLEN;
/* checksum rsrc fork data */
if ( afinfo->as_ents[ AS_RFE ].ae_length > 0 ) {
if ( snprintf( rsrc_path, MAXPATHLEN, "%s%s",
path, _PATH_RSRCFORKSPEC ) >= MAXPATHLEN ) {
errno = ENAMETOOLONG;
return( -1 );
}
if (( rfd = open( rsrc_path, O_RDONLY )) < 0 ) {
return( -1 );
}
while (( rc = read( rfd, buf, sizeof( buf ))) > 0 ) {
EVP_DigestUpdate( mdctx, buf, (unsigned int)rc );
size += (size_t)rc;
}
if ( close( rfd ) < 0 ) {
return( -1 );
}
if ( rc < 0 ) {
return( -1 );
}
}
if (( dfd = open( path, O_RDONLY, 0 )) < 0 ) {
return( -1 );
}
/* checksum data fork */
while (( rc = read( dfd, buf, sizeof( buf ))) > 0 ) {
EVP_DigestUpdate( mdctx, buf, (unsigned int)rc );
size += (size_t)rc;
}
if ( rc < 0 ) {
return( -1 );
}
if ( close( dfd ) < 0 ) {
return( -1 );
}
EVP_DigestFinal( mdctx, md_value, &md_len );
base64_e( ( unsigned char* ) md_value, md_len, cksum_b64 );
EVP_MD_CTX_free( mdctx );
return( size );
}
#else /* __APPLE__ */
/*
* stub fuction for non-hfs+ machines.
*
* return values:
* -1 system error: non hfs+ system
*/
off_t
do_acksum( char *path, char *cksum_b64, struct applefileinfo *afino )
{
errno = EOPNOTSUPP;
return( -1 );
}
#endif /* __APPLE__ */
#ifdef ENABLE_XATTR
off_t
do_xcksum( char *path, char *cksum_b64, char *xname )
{
ssize_t xr;
unsigned int offset = 0;
char *buf = NULL;
unsigned int md_len;
extern EVP_MD *md;
EVP_MD_CTX *mdctx = EVP_MD_CTX_new();
unsigned char md_value[ SZ_BASE64_D( SZ_BASE64_E( EVP_MAX_MD_SIZE ) ) ];
EVP_DigestInit( mdctx, md );
/*
* For now we're skipping Finder Info and Resource Fork, since our
* AppleFile code is handling them. Apple's getxattr call can take
* an offset, which would allow something like looping reads, but
* the offset "is only used with the resource fork attribute,"
* according to the manpage. "For all other extended attributes,
* this parameter is reserved and should be zero." Gah.
*/
if (( xr = xattr_get( path, xname, &buf, offset )) < 0 ) {
fprintf( stderr, "xattr_get failed: %s, %s\n", path, xname );
return( -1 );
}
EVP_DigestUpdate( mdctx, buf, (unsigned int)xr );
EVP_DigestFinal( mdctx, md_value, &md_len );
base64_e( md_value, md_len, cksum_b64 );
return((off_t)xr );
}
#endif /* ENABLE_XATTR */