forked from carbon-language/carbon-lang
-
Notifications
You must be signed in to change notification settings - Fork 0
/
parse.cpp
90 lines (74 loc) · 2.65 KB
/
parse.cpp
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
// Part of the Carbon Language project, under the Apache License v2.0 with LLVM
// Exceptions. See /LICENSE for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
#include "explorer/syntax/parse.h"
#include "common/check.h"
#include "common/error.h"
#include "explorer/common/error_builders.h"
#include "explorer/syntax/lexer.h"
#include "explorer/syntax/parse_and_lex_context.h"
#include "explorer/syntax/parser.h"
#include "llvm/Support/Error.h"
namespace Carbon {
static auto ParseImpl(yyscan_t scanner, Nonnull<Arena*> arena,
std::string_view input_file_name, bool parser_debug)
-> ErrorOr<AST> {
// Prepare other parser arguments.
std::optional<AST> ast = std::nullopt;
ParseAndLexContext context(arena->New<std::string>(input_file_name),
parser_debug);
// Do the parse.
auto parser = Parser(arena, scanner, context, &ast);
if (parser_debug) {
parser.set_debug_level(1);
}
if (auto syntax_error_code = parser(); syntax_error_code != 0) {
auto errors = context.take_errors();
if (errors.empty()) {
return Error("Unknown parser erroor");
}
return std::move(errors.front());
}
// Return parse results.
CARBON_CHECK(ast != std::nullopt)
<< "parser validated syntax yet didn't produce an AST.";
return *ast;
}
auto Parse(Nonnull<Arena*> arena, std::string_view input_file_name,
bool parser_debug) -> ErrorOr<AST> {
std::string name_str(input_file_name);
FILE* input_file = fopen(name_str.c_str(), "r");
if (input_file == nullptr) {
return ProgramError(SourceLocation(name_str.c_str(), 0))
<< "Error opening file: " << std::strerror(errno);
}
// Prepare the lexer.
yyscan_t scanner;
yylex_init(&scanner);
auto buffer = yy_create_buffer(input_file, YY_BUF_SIZE, scanner);
yy_switch_to_buffer(buffer, scanner);
ErrorOr<AST> result =
ParseImpl(scanner, arena, input_file_name, parser_debug);
// Clean up the lexer.
yy_delete_buffer(buffer, scanner);
yylex_destroy(scanner);
fclose(input_file);
return result;
}
auto ParseFromString(Nonnull<Arena*> arena, std::string_view input_file_name,
std::string_view file_contents, bool parser_debug)
-> ErrorOr<AST> {
// Prepare the lexer.
yyscan_t scanner;
yylex_init(&scanner);
auto buffer =
yy_scan_bytes(file_contents.data(), file_contents.size(), scanner);
yy_switch_to_buffer(buffer, scanner);
ErrorOr<AST> result =
ParseImpl(scanner, arena, input_file_name, parser_debug);
// Clean up the lexer.
yy_delete_buffer(buffer, scanner);
yylex_destroy(scanner);
return result;
}
} // namespace Carbon