blob: 8e5adcad5132de6c06d5e1240e0931c0656fab52 [file] [log] [blame]
Chris Lattner4b009652007-07-25 00:24:17 +00001//===--- ASTStreamer.cpp - Provide streaming interface to ASTs ------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file was developed by Chris Lattner and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the ASTStreamer interface.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/Sema/ASTStreamer.h"
15#include "clang/AST/ASTContext.h"
Chris Lattner1cc01712007-09-15 22:56:56 +000016#include "clang/AST/ASTConsumer.h"
Chris Lattner4b009652007-07-25 00:24:17 +000017#include "Sema.h"
18#include "clang/Parse/Action.h"
19#include "clang/Parse/Parser.h"
20using namespace clang;
21
Chris Lattner1cc01712007-09-15 22:56:56 +000022ASTConsumer::~ASTConsumer() {}
23
Chris Lattner4b009652007-07-25 00:24:17 +000024namespace {
25 class ASTStreamer {
26 Parser P;
27 std::vector<Decl*> LastInGroupList;
28 public:
29 ASTStreamer(Preprocessor &pp, ASTContext &ctxt, unsigned MainFileID)
30 : P(pp, *new Sema(pp, ctxt, LastInGroupList)) {
31 pp.EnterSourceFile(MainFileID, 0, true);
32
33 // Initialize the parser.
34 P.Initialize();
35 }
36
37 /// ReadTopLevelDecl - Parse and return the next top-level declaration.
38 Decl *ReadTopLevelDecl();
39
40 void PrintStats() const;
41
42 ~ASTStreamer() {
43 P.Finalize();
44 delete &P.getActions();
45 }
46 };
47}
48
49/// ReadTopLevelDecl - Parse and return the next top-level declaration.
50///
51Decl *ASTStreamer::ReadTopLevelDecl() {
52 Parser::DeclTy *Result;
53
54 /// If the previous time through we read something like 'int X, Y', return
55 /// the next declarator.
56 if (!LastInGroupList.empty()) {
57 Result = LastInGroupList.back();
58 LastInGroupList.pop_back();
59 return static_cast<Decl*>(Result);
60 }
61
62 do {
63 if (P.ParseTopLevelDecl(Result))
64 return 0; // End of file.
65
66 // If we got a null return and something *was* parsed, try again. This
67 // is due to a top-level semicolon, an action override, or a parse error
68 // skipping something.
69 } while (Result == 0);
70
71 // If we parsed a declspec with multiple declarators, reverse the list and
72 // return the first one.
73 if (!LastInGroupList.empty()) {
74 LastInGroupList.push_back((Decl*)Result);
75 std::reverse(LastInGroupList.begin(), LastInGroupList.end());
76 Result = LastInGroupList.back();
77 LastInGroupList.pop_back();
78 }
79
80 return static_cast<Decl*>(Result);
81}
82
83void ASTStreamer::PrintStats() const {
84}
85
86//===----------------------------------------------------------------------===//
87// Public interface to the file
88//===----------------------------------------------------------------------===//
89
Chris Lattner1cc01712007-09-15 22:56:56 +000090/// ParseAST - Parse the entire file specified, notifying the ASTConsumer as
91/// the file is parsed.
92void clang::ParseAST(Preprocessor &PP, unsigned MainFileID,
93 ASTConsumer &Consumer, bool PrintStats) {
94 // Collect global stats on Decls/Stmts (until we have a module streamer).
95 if (PrintStats) {
96 Decl::CollectingStats(true);
97 Stmt::CollectingStats(true);
98 }
99
100 ASTContext Context(PP.getSourceManager(), PP.getTargetInfo(),
Steve Naroff4ed9d662007-09-27 14:38:14 +0000101 PP.getIdentifierTable(), PP.getSelectorTable());
Chris Lattner1cc01712007-09-15 22:56:56 +0000102
103 ASTStreamer Streamer(PP, Context, MainFileID);
104
105 Consumer.Initialize(Context, MainFileID);
106
107 while (Decl *D = Streamer.ReadTopLevelDecl())
108 Consumer.HandleTopLevelDecl(D);
109
110 if (PrintStats) {
111 fprintf(stderr, "\nSTATISTICS:\n");
112 Streamer.PrintStats();
113 Context.PrintStats();
114 Decl::PrintStats();
115 Stmt::PrintStats();
116 Consumer.PrintStats();
117
118 Decl::CollectingStats(false);
119 Stmt::CollectingStats(false);
120 }
121}