/* -*- mode: objc; coding: utf-8 -*- */ /* Toilet Lisp, a Common Lisp subset for the Étoilé runtime. * Copyright (C) 2008 Matthias Andreas Benkard. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or (at * your option) any later version. * * This program 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 * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #import "MLKReadtable.h" #import "MLKCharacter.h" #import "runtime-compatibility.h" #import #import @implementation MLKReadtable -(MLKReadtable *) initSuper { return [super init]; } -(MLKReadtable *) init { self = [super init]; _syntaxTable = [[NSMutableDictionary alloc] init]; _readerMacros = [[NSMutableDictionary alloc] init]; _traits = [[NSMutableDictionary alloc] init]; _case = MLKReadtableCase_UPCASE; return self; } -(MLKReadtable *) copyWithZone:(NSZone *)zone { MLKReadtable *copy = [[MLKReadtable allocWithZone:zone] initSuper]; copy->_syntaxTable = [_syntaxTable mutableCopyWithZone:zone]; copy->_readerMacros = [_readerMacros mutableCopyWithZone:zone]; copy->_traits = [_traits mutableCopyWithZone:zone]; copy->_case = _case; return copy; } -(BOOL) isWhitespaceCharacter:(unichar)ch { return ([[_syntaxTable objectForKey:[NSNumber numberWithLong:ch]] isEqual:[NSNumber numberWithShort:WHITESPACE]]); } -(BOOL) isMacroCharacter:(unichar)ch; { return ([self isNonTerminatingMacroCharacter:ch] || [self isTerminatingMacroCharacter:ch]); } -(enum MLKSyntaxType) characterSyntaxType:(unichar)ch { NSNumber *type = [_syntaxTable objectForKey:[NSNumber numberWithLong:ch]]; if (!type) return CONSTITUENT; else return [type intValue]; } #define DEFINE_SYNTAX_PREDICATE(SELECTOR, SYNTAX_TYPE) \ -(BOOL) SELECTOR (unichar)ch \ { \ return ([self characterSyntaxType:ch] \ == SYNTAX_TYPE); \ } DEFINE_SYNTAX_PREDICATE(isNonTerminatingMacroCharacter:, NONTERMINATING_MACRO) DEFINE_SYNTAX_PREDICATE(isTerminatingMacroCharacter:, TERMINATING_MACRO) DEFINE_SYNTAX_PREDICATE(isSingleEscapeCharacter:, SINGLE_ESCAPE) DEFINE_SYNTAX_PREDICATE(isMultipleEscapeCharacter:, MULTI_ESCAPE) DEFINE_SYNTAX_PREDICATE(isConstituentCharacter:, CONSTITUENT) -(BOOL) characterHasCase:(unichar)ch { return (![[[NSString stringWithFormat:@"%C", ch] uppercaseString] isEqual:[[NSString stringWithFormat:@"%C", ch] lowercaseString]]); } -(id ) macroFunctionForCharacter:(unichar)ch; { return [_readerMacros objectForKey:[NSNumber numberWithLong:ch]]; } -(void) setMacroFunction:(id )function forCharacter:(unichar)ch { [_readerMacros setObject:function forKey:[NSNumber numberWithLong:ch]]; } -(unichar) charWithReadtableCase:(unichar)ch { switch (_case) { case MLKReadtableCase_PRESERVE: return ch; case MLKReadtableCase_UPCASE: return [MLKCharacter uppercaseCharForChar:ch]; case MLKReadtableCase_DOWNCASE: return [MLKCharacter lowercaseCharForChar:ch]; case MLKReadtableCase_INVERT: { unichar upCh; upCh = [MLKCharacter uppercaseCharForChar:ch]; if (ch == upCh) return [MLKCharacter lowercaseCharForChar:ch]; else return upCh; } } return 0; } -(int) characterConstituentTraits:(unichar)ch { NSNumber *traits = [_traits objectForKey:[NSNumber numberWithLong:ch]]; if (!traits) return ALPHABETIC; else return [traits intValue]; } -(BOOL) character:(unichar)ch hasTrait:(enum MLKConstituentTrait)trait { int traits = [self characterConstituentTraits:ch]; return (traits & trait) != 0; } #define DEFINE_TRAIT_PREDICATE(SELECTOR, TRAIT) \ -(BOOL) SELECTOR (unichar)ch \ { \ return ([self character:ch hasTrait:TRAIT]); \ } DEFINE_TRAIT_PREDICATE(isInvalid:, INVALID) DEFINE_TRAIT_PREDICATE(isAlphabetic:, ALPHABETIC) DEFINE_TRAIT_PREDICATE(isPackageMarker:, PACKAGE_MARKER) DEFINE_TRAIT_PREDICATE(isAlphaDigit:, ALPHA_DIGIT) DEFINE_TRAIT_PREDICATE(isExponentMarker:, EXPONENT_MARKER) DEFINE_TRAIT_PREDICATE(isNumberMarker:, NUMBER_MARKER) DEFINE_TRAIT_PREDICATE(isRatioMarker:, RATIO_MARKER) DEFINE_TRAIT_PREDICATE(isDecimalPoint:, DECIMAL_POINT) DEFINE_TRAIT_PREDICATE(isMinusSign:, MINUS_SIGN) DEFINE_TRAIT_PREDICATE(isPlusSign:, PLUS_SIGN) DEFINE_TRAIT_PREDICATE(isSign:, SIGN) DEFINE_TRAIT_PREDICATE(isDot:, DOT) -(void) setSyntaxType:(enum MLKSyntaxType)type forCharacter:(unichar)ch { [_syntaxTable setObject:[NSNumber numberWithInt:type] forKey:[NSNumber numberWithLong:ch]]; } -(void) setConstituentTrait:(enum MLKConstituentTrait)trait forCharacter:(unichar)ch { int traits = [self characterConstituentTraits:ch]; traits = traits | trait; [_traits setObject:[NSNumber numberWithInt:traits] forKey:[NSNumber numberWithLong:ch]]; } -(void) unsetConstituentTrait:(enum MLKConstituentTrait)trait forCharacter:(unichar)ch { int traits = [self characterConstituentTraits:ch]; traits = traits & ~trait; [_traits setObject:[NSNumber numberWithInt:traits] forKey:[NSNumber numberWithLong:ch]]; } -(BOOL) isDecimalDigit:(unichar)ch { return [self isDigit:ch inBase:10]; } -(BOOL) isDigit:(unichar)ch inBase:(int)base { if (base < 11) return ('0' <= ch && ch < '0' + base); else return (('0' <= ch && ch <= '9') || ('A' <= ch && ch < 'A' + base - 10) || ('a' <= ch && ch < 'a' + base - 10)); } -(int) digitWeight:(unichar)ch { if ('0' <= ch && ch <= '9') return (ch - '0'); else if ('A' <= ch && ch <= 'Z') return (ch - 'A' + 10); else return (ch - 'a' + 10); } @end