/* * Copyright (c) 2008 - 2009 * Jonathan Schleifer <js@webkeks.org> * * All rights reserved. * * This file is part of libobjfw. It may be distributed under the terms of the * Q Public License 1.0, which can be found in the file LICENSE included in * the packaging of this file. */ #import "config.h" #include <string.h> #include <unistd.h> #import "OFStream.h" #import "OFExceptions.h" #import "OFMacros.h" extern int getpagesize(void); @implementation OFStream - init { self = [super init]; cache = NULL; return self; } - (size_t)readNBytes: (size_t)size intoBuffer: (char*)buf { @throw [OFNotImplementedException newWithClass: isa andSelector: _cmd]; } - (OFString*)readLine { size_t i, len; char *ret_c, *tmp, *tmp2; OFString *ret; /* Look if there's a line or \0 in our cache */ if (cache != NULL) { for (i = 0; i < cache_len; i++) { if (OF_UNLIKELY(cache[i] == '\n' || cache[i] == '\0')) { ret_c = [self allocWithSize: i + 1]; memcpy(ret_c, cache, i); ret_c[i] = '\0'; @try { tmp = [self allocWithSize: cache_len - i - 1]; } @catch (OFException *e) { [self freeMem: ret_c]; @throw e; } memcpy(tmp, cache + i + 1, cache_len - i - 1); [self freeMem: cache]; cache = tmp; cache_len = cache_len - i - 1; @try { ret = [OFString stringWithCString: ret_c]; } @finally { [self freeMem: ret_c]; } return ret; } } } /* Read until we get a newline or \0 */ tmp = [self allocWithSize: getpagesize()]; for (;;) { @try { len = [self readNBytes: getpagesize() - 1 intoBuffer: tmp]; } @catch (OFException *e) { [self freeMem: tmp]; @throw e; } /* Look if there's a newline or \0 */ for (i = 0; i < len; i++) { if (OF_UNLIKELY(tmp[i] == '\n' || tmp[i] == '\0')) { @try { ret_c = [self allocWithSize: cache_len + i + 1]; } @catch (OFException *e) { [self freeMem: tmp]; @throw e; } if (cache != NULL) memcpy(ret_c, cache, cache_len); memcpy(ret_c + cache_len, tmp, i); ret_c[i] = '\0'; if (i < len) { @try { tmp2 = [self allocWithSize: len - i - 1]; } @catch (OFException *e) { [self freeMem: ret_c]; [self freeMem: tmp]; @throw e; } memcpy(tmp2, tmp + i + 1, len - i - 1); if (cache != NULL) [self freeMem: cache]; cache = tmp2; cache_len = len - i - 1; } else { if (cache != NULL) [self freeMem: cache]; cache = NULL; cache_len = 0; } [self freeMem: tmp]; @try { ret = [OFString stringWithCString: ret_c]; } @finally { [self freeMem: ret_c]; } return ret; } } /* There was no newline or \0 */ @try { cache = [self resizeMem: cache toSize: cache_len + len]; } @catch (OFException *e) { [self freeMem: tmp]; @throw e; } memcpy(cache + cache_len, tmp, len); cache_len += len; } } - (size_t)writeNBytes: (size_t)size fromBuffer: (const char*)buf { @throw [OFNotImplementedException newWithClass: isa andSelector: _cmd]; } - (size_t)writeString: (OFString*)str { return [self writeNBytes: [str length] fromBuffer: [str cString]]; } - (size_t)getCache: (char**)ptr { if (ptr != NULL) *ptr = cache; return cache_len; } - clearCache { if (cache != NULL) [self freeMem: cache]; cache = NULL; cache_len = 0; return self; } - close { @throw [OFNotImplementedException newWithClass: isa andSelector: _cmd]; } @end