XADMaster/XADSkipHandle.m
2018-03-21 14:24:50 +02:00

230 lines
5.4 KiB
Objective-C

/*
* XADSkipHandle.m
*
* Copyright (c) 2017-present, MacPaw Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
* MA 02110-1301 USA
*/
#import "XADSkipHandle.h"
#import "Realloc.h"
@implementation XADSkipHandle
static off_t ActualStart(XADSkipHandle *self,int index)
{
return self->regions[index].actual;
}
static off_t ActualGapStart(XADSkipHandle *self,int index)
{
if(index>=self->numregions-1) return CSHandleMaxLength;
return self->regions[index].actual+self->regions[index+1].skip-self->regions[index].skip;
}
static off_t ActualEnd(XADSkipHandle *self,int index)
{
if(index>=self->numregions-1) return CSHandleMaxLength;
return self->regions[index+1].actual;
}
static off_t SkipStart(XADSkipHandle *self,int index)
{
return self->regions[index].skip;
}
static off_t SkipEnd(XADSkipHandle *self,int index)
{
if(index>=self->numregions-1) return CSHandleMaxLength;
return self->regions[index+1].skip;
}
static int FindIndexOfRegionContainingActualOffset(XADSkipHandle *self,off_t pos)
{
int first=0,last=self->numregions-1;
if(ActualStart(self,last)<=pos) return last;
if(ActualEnd(self,first)>pos) return first;
while(last-first>1)
{
int mid=(last+first)/2;
if(ActualStart(self,mid)<=pos)
{
if(ActualEnd(self,mid)>pos) return mid;
first=mid;
}
else last=mid;
}
return first;
}
static int FindIndexOfRegionContainingSkipOffset(XADSkipHandle *self,off_t pos)
{
int first=0,last=self->numregions-1;
if(SkipStart(self,last)<=pos) return last;
if(SkipEnd(self,first)>pos) return first;
while(last-first>1)
{
int mid=(last+first)/2;
if(SkipStart(self,mid)<=pos)
{
if(SkipEnd(self,mid)>pos) return mid;
first=mid;
}
else last=mid;
}
return first;
}
static off_t SkipOffsetToActual(XADSkipHandle *self,off_t pos)
{
int index=FindIndexOfRegionContainingSkipOffset(self,pos);
return pos+ActualStart(self,index)-SkipStart(self,index);
}
static off_t ActualOffsetToSkip(XADSkipHandle *self,off_t pos)
{
int index=FindIndexOfRegionContainingActualOffset(self,pos);
// Skip ahead to next region if position is in the gap between regions
if(pos>=ActualGapStart(self,index)) return SkipStart(self,index+1);
return pos-ActualStart(self,index)+SkipStart(self,index);
}
-(id)initWithHandle:(CSHandle *)handle
{
if((self=[super initWithParentHandle:handle]))
{
regions=malloc(sizeof(XADSkipRegion));
regions[0].actual=regions[0].skip=0;
numregions=1;
}
return self;
}
-(id)initAsCopyOf:(XADSkipHandle *)other
{
if((self=[super initAsCopyOf:other]))
{
numregions=other->numregions;
regions=malloc(sizeof(XADSkipRegion)*numregions);
memcpy(regions,other->regions,sizeof(XADSkipRegion)*numregions);
}
return self;
}
-(void)dealloc
{
free(regions);
[super dealloc];
}
-(void)addSkipFrom:(off_t)start length:(off_t)length
{
[self addSkipFrom:start to:start+length];
}
-(void)addSkipFrom:(off_t)start to:(off_t)end
{
int index=FindIndexOfRegionContainingActualOffset(self,start);
// TODO: merge regions instead of bailing out
if(end>=ActualGapStart(self,index)) [NSException raise:NSInvalidArgumentException format:@"Attempted to add overlapping or neighbouring skips"];
regions=Realloc(regions,sizeof(XADSkipRegion)*(numregions+1));
for(int i=numregions-1;i>index;i++)
{
regions[i+1].actual=regions[i].actual;
regions[i+1].skip=regions[i].skip-end+start;
}
regions[index+1].actual=end;
regions[index+1].skip=regions[index].skip+start-regions[index].actual;
numregions++;
}
-(off_t)actualOffsetForSkipOffset:(off_t)skipoffset { return SkipOffsetToActual(self,skipoffset); }
-(off_t)skipOffsetForActualOffset:(off_t)actualoffset { return ActualOffsetToSkip(self,actualoffset); }
-(off_t)fileSize
{
off_t size=[parent fileSize];
if(size==CSHandleMaxLength) return CSHandleMaxLength;
return ActualOffsetToSkip(self,size-1)+1;
}
-(off_t)offsetInFile
{
off_t offs=[parent offsetInFile];
return ActualOffsetToSkip(self,offs);
}
-(BOOL)atEndOfFile
{
return [super atEndOfFile];
// TODO: handle skips at EOF
}
-(void)seekToFileOffset:(off_t)offs
{
[parent seekToFileOffset:SkipOffsetToActual(self,offs)];
}
-(void)seekToEndOfFile
{
[parent seekToEndOfFile];
// TODO: handle skips at EOF
}
-(int)readAtMost:(int)num toBuffer:(void *)buffer
{
off_t pos=[parent offsetInFile];
int index=FindIndexOfRegionContainingActualOffset(self,pos);
if(pos>=ActualGapStart(self,index)) [parent seekToFileOffset:pos=ActualStart(self,++index)];
int total=0;
for(;;)
{
off_t gap=ActualGapStart(self,index);
if(num-total<gap-pos)
{
total+=[parent readAtMost:num-total toBuffer:buffer+total];
return total;
}
int actual=[parent readAtMost:(int)(gap-pos) toBuffer:buffer+total];
total+=actual;
if(actual!=gap-pos) return total;
[parent seekToFileOffset:pos=ActualStart(self,++index)];
}
}
@end