-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathPDFEncryptionUtils.m
121 lines (93 loc) · 2.65 KB
/
PDFEncryptionUtils.m
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
#import "PDFEncryptionUtils.h"
NSString *PDFMD5FinishedException=@"PDFMD5FinishedException";
@implementation PDFMD5Engine
+(PDFMD5Engine *)engine { return [[[self class] new] autorelease]; }
+(NSData *)digestForData:(NSData *)data { return [self digestForBytes:[data bytes] length:[data length]]; }
+(NSData *)digestForBytes:(const void *)bytes length:(int)length
{
PDFMD5Engine *md5=[[self class] new];
[md5 updateWithBytes:bytes length:length];
NSData *res=[md5 digest];
[md5 release];
return res;
}
-(id)init
{
if(self=[super init])
{
MD5_Init(&md5);
done=NO;
}
return self;
}
-(void)updateWithData:(NSData *)data { [self updateWithBytes:[data bytes] length:[data length]]; }
-(void)updateWithBytes:(const void *)bytes length:(unsigned long)length
{
if(done) [NSException raise:PDFMD5FinishedException format:@"Attempted to update a finished %@ object",[self class]];
MD5_Update(&md5,bytes,length);
}
-(NSData *)digest
{
if(!done) { MD5_Final(digest_bytes,&md5); done=YES; }
return [NSData dataWithBytes:digest_bytes length:16];
}
-(NSString *)hexDigest
{
if(!done) { MD5_Final(digest_bytes,&md5); done=YES; }
return [NSString stringWithFormat:@"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
digest_bytes[0],digest_bytes[1],digest_bytes[2],digest_bytes[3],
digest_bytes[4],digest_bytes[5],digest_bytes[6],digest_bytes[7],
digest_bytes[8],digest_bytes[9],digest_bytes[10],digest_bytes[11],
digest_bytes[12],digest_bytes[13],digest_bytes[14],digest_bytes[15]];
}
-(NSString *)description
{
if(done) return [NSString stringWithFormat:@"<%@ with digest %@>",[self class],[self hexDigest]];
else return [NSString stringWithFormat:@"<%@, unfinished>",[self class]];
}
@end
@implementation PDFAESHandle
-(id)initWithHandle:(CSHandle *)handle key:(NSData *)keydata
{
if(self=[super initWithName:[handle name]])
{
parent=[handle retain];
key=[keydata retain];
iv=[parent copyDataOfLength:16];
startoffs=[parent offsetInFile];
[self setBlockPointer:streambuffer];
AES_set_decrypt_key([key bytes],[key length]*8,&aeskey);
}
return self;
}
-(void)dealloc
{
[parent release];
[key release];
[iv release];
[super dealloc];
}
-(void)resetBlockStream
{
[parent seekToFileOffset:startoffs];
memcpy(ivbuffer,[iv bytes],16);
}
-(int)produceBlockAtOffset:(off_t)pos
{
uint8_t inbuf[16];
[parent readBytes:16 toBuffer:inbuf];
AES_cbc_encrypt(inbuf,streambuffer,16,&aeskey,ivbuffer,AES_DECRYPT);
if([parent atEndOfFile])
{
[self endBlockStream];
int val=streambuffer[15];
if(val>0&&val<=16)
{
for(int i=1;i<val;i++) if(streambuffer[15-val]!=val) return 0;
return 16-val;
}
else return 0;
}
else return 16;
}
@end