mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-24 17:51:22 +00:00
116 lines
2.6 KiB
R
116 lines
2.6 KiB
R
#include "Cons.h"
|
|
#include "Parser.h"
|
|
#include "Nil.h"
|
|
#include "Error.h"
|
|
#include "defs.h"
|
|
|
|
@implementation Parser
|
|
|
|
+ (id) newFromSource: (string) s file: (string) f
|
|
{
|
|
return [[self alloc] initWithSource: s file: f];
|
|
}
|
|
|
|
- (id) initWithSource: (string) s file: (string) f
|
|
{
|
|
self = [super init];
|
|
lexer = [Lexer newFromSource: s file: f];
|
|
file = f;
|
|
return self;
|
|
}
|
|
|
|
- (SchemeObject*) readList
|
|
{
|
|
local SchemeObject *token, *res;
|
|
local int line;
|
|
local Error *err;
|
|
|
|
line = [lexer lineNumber];
|
|
token = [self readAtomic];
|
|
|
|
if ([token isError])
|
|
return token;
|
|
|
|
if (!token) {
|
|
err = [Error type: "parse" message: "Unmatched open parenthesis"];
|
|
[err source: file];
|
|
[err line: [lexer lineNumber]];
|
|
return err;
|
|
}
|
|
|
|
if (token == [Symbol rightParen]) {
|
|
return [Nil nil];
|
|
} else if (token == [Symbol dot]) {
|
|
res = [self readAtomic];
|
|
if ([res isError]) return res;
|
|
if ([self readAtomic] != [Symbol rightParen]) {
|
|
err = [Error type: "parse" message: "Improper use of dot"];
|
|
[err source: file];
|
|
[err line: [lexer lineNumber]];
|
|
return err;
|
|
}
|
|
return res;
|
|
} else {
|
|
res = [self readList];
|
|
if ([res isError]) return res;
|
|
res = cons(token, res);
|
|
[res source: file];
|
|
[res line: line];
|
|
return res;
|
|
}
|
|
}
|
|
|
|
- (SchemeObject*) readAtomic
|
|
{
|
|
local SchemeObject *token, *list, *res;
|
|
local int line;
|
|
|
|
line = [lexer lineNumber];
|
|
|
|
token = [lexer nextToken];
|
|
|
|
if ([token isError]) {
|
|
return token;
|
|
}
|
|
|
|
if (!token) {
|
|
return nil;
|
|
}
|
|
|
|
if (token == [Symbol leftParen]) {
|
|
list = [self readList];
|
|
return list;
|
|
} else if (token == [Symbol quote]) {
|
|
res = [self read];
|
|
if ([res isError]) return res;
|
|
res = cons(res, [Nil nil]);
|
|
[res source: file];
|
|
[res line: line];
|
|
res = cons([Symbol forString: "quote"], res);
|
|
[res source: file];
|
|
[res line: line];
|
|
return res;
|
|
} else return token;
|
|
}
|
|
|
|
- (SchemeObject*) read
|
|
{
|
|
local SchemeObject *token;
|
|
local Error *err;
|
|
|
|
token = [self readAtomic];
|
|
if (token == [Symbol rightParen]) {
|
|
err = [Error type: "parse" message: "mismatched close parentheis"];
|
|
[err source: file];
|
|
[err line: [lexer lineNumber]];
|
|
return err;
|
|
}
|
|
return token;
|
|
}
|
|
|
|
- (void) markReachable
|
|
{
|
|
[lexer mark];
|
|
}
|
|
|
|
@end
|