File: XADDigestHandle.m

package info (click to toggle)
unar 1.1-2
  • links: PTS, VCS
  • area: main
  • in suites: wheezy
  • size: 6,664 kB
  • sloc: ansic: 52,939; objc: 39,563; cpp: 4,074; makefile: 99; perl: 10
file content (96 lines) | stat: -rw-r--r-- 2,355 bytes parent folder | download
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
#import "XADDigestHandle.h"

@implementation XADDigestHandle

+(XADDigestHandle *)MD5HandleWithHandle:(CSHandle *)handle length:(off_t)length
correctDigest:(NSData *)correctdigest
{
	return [[[self alloc] initWithHandle:handle length:length digestType:EVP_md5() correctDigest:correctdigest] autorelease];
}

+(XADDigestHandle *)SHA1HandleWithHandle:(CSHandle *)handle length:(off_t)length
correctDigest:(NSData *)correctdigest
{
	return [[[self alloc] initWithHandle:handle length:length digestType:EVP_sha1() correctDigest:correctdigest] autorelease];
}

/*+(XADDigestHandle *)SHA256HandleWithHandle:(CSHandle *)handle length:(off_t)length
correctDigest:(NSData *)correctdigest
{
	return [[[self alloc] initWithHandle:handle length:length digestType:EVP_ssha256() correctDigest:correctdigest] autorelease];
}*/

+(XADDigestHandle *)digestHandleWithHandle:(CSHandle *)handle length:(off_t)length
digestName:(NSString *)digestname correctDigest:(NSData *)correctdigest
{
	static BOOL added=NO;
	if(!added)
	{
		OpenSSL_add_all_digests();
		added=YES;
	}

	const EVP_MD *type=EVP_get_digestbyname([digestname UTF8String]);
	if(!type) return nil;

	return [[[self alloc] initWithHandle:handle length:length digestType:type correctDigest:correctdigest] autorelease];
}

-(id)initWithHandle:(CSHandle *)handle length:(off_t)length
digestType:(const EVP_MD *)digesttype correctDigest:(NSData *)correctdigest
{
	if((self=[super initWithName:[handle name] length:length]))
	{
		parent=[handle retain];
		digest=[correctdigest retain];
		type=digesttype;
		inited=NO;
	}
	return self;
}

-(void)dealloc
{
	if(inited) EVP_MD_CTX_cleanup(&ctx);
	[parent release];
	[digest release];
	[super dealloc];
}

-(void)resetStream
{
	if(inited) EVP_MD_CTX_cleanup(&ctx);
	EVP_DigestInit(&ctx,type);
	inited=YES;

	[parent seekToFileOffset:0];
}

-(int)streamAtMost:(int)num toBuffer:(void *)buffer
{
	int actual=[parent readAtMost:num toBuffer:buffer];
	EVP_DigestUpdate(&ctx,buffer,actual);
	return actual;
}

-(BOOL)hasChecksum { return YES; }

-(BOOL)isChecksumCorrect
{
	int length=EVP_MD_size(type);
	if([digest length]!=length) return NO;

	EVP_MD_CTX copy;
	EVP_MD_CTX_copy(&copy,&ctx);

	uint8_t buf[length];
	EVP_DigestFinal(&ctx,buf,NULL);

	return memcmp([digest bytes],buf,length)==0;
}

-(double)estimatedProgress { return [parent estimatedProgress]; }

@end