blob: edcde871a038cb180ab314b3c094c616f7415270 [file] [log] [blame]
Argyrios Kyrtzidisf3d587e2012-12-04 07:27:05 +00001//===--- PPConditionalDirectiveRecord.h - Preprocessing Directives-*- C++ -*-=//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the PPConditionalDirectiveRecord class, which maintains
11// a record of conditional directive regions.
12//
13//===----------------------------------------------------------------------===//
14#include "clang/Lex/PPConditionalDirectiveRecord.h"
15#include "llvm/Support/Capacity.h"
16
17using namespace clang;
18
19PPConditionalDirectiveRecord::PPConditionalDirectiveRecord(SourceManager &SM)
20 : SourceMgr(SM) {
21 CondDirectiveStack.push_back(SourceLocation());
22}
23
24bool PPConditionalDirectiveRecord::rangeIntersectsConditionalDirective(
25 SourceRange Range) const {
26 if (Range.isInvalid())
27 return false;
28
29 CondDirectiveLocsTy::const_iterator
30 low = std::lower_bound(CondDirectiveLocs.begin(), CondDirectiveLocs.end(),
31 Range.getBegin(), CondDirectiveLoc::Comp(SourceMgr));
32 if (low == CondDirectiveLocs.end())
33 return false;
34
35 if (SourceMgr.isBeforeInTranslationUnit(Range.getEnd(), low->getLoc()))
36 return false;
37
38 CondDirectiveLocsTy::const_iterator
39 upp = std::upper_bound(low, CondDirectiveLocs.end(),
40 Range.getEnd(), CondDirectiveLoc::Comp(SourceMgr));
41 SourceLocation uppRegion;
42 if (upp != CondDirectiveLocs.end())
43 uppRegion = upp->getRegionLoc();
44
45 return low->getRegionLoc() != uppRegion;
46}
47
48SourceLocation PPConditionalDirectiveRecord::findConditionalDirectiveRegionLoc(
49 SourceLocation Loc) const {
50 if (Loc.isInvalid())
51 return SourceLocation();
52 if (CondDirectiveLocs.empty())
53 return SourceLocation();
54
55 if (SourceMgr.isBeforeInTranslationUnit(CondDirectiveLocs.back().getLoc(),
56 Loc))
57 return CondDirectiveStack.back();
58
59 CondDirectiveLocsTy::const_iterator
60 low = std::lower_bound(CondDirectiveLocs.begin(), CondDirectiveLocs.end(),
61 Loc, CondDirectiveLoc::Comp(SourceMgr));
62 assert(low != CondDirectiveLocs.end());
63 return low->getRegionLoc();
64}
65
66void PPConditionalDirectiveRecord::addCondDirectiveLoc(
67 CondDirectiveLoc DirLoc) {
68 // Ignore directives in system headers.
69 if (SourceMgr.isInSystemHeader(DirLoc.getLoc()))
70 return;
71
72 assert(CondDirectiveLocs.empty() ||
73 SourceMgr.isBeforeInTranslationUnit(CondDirectiveLocs.back().getLoc(),
74 DirLoc.getLoc()));
75 CondDirectiveLocs.push_back(DirLoc);
76}
77
78void PPConditionalDirectiveRecord::If(SourceLocation Loc,
79 SourceRange ConditionRange) {
80 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
81 CondDirectiveStack.push_back(Loc);
82}
83
84void PPConditionalDirectiveRecord::Ifdef(SourceLocation Loc,
85 const Token &MacroNameTok) {
86 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
87 CondDirectiveStack.push_back(Loc);
88}
89
90void PPConditionalDirectiveRecord::Ifndef(SourceLocation Loc,
91 const Token &MacroNameTok) {
92 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
93 CondDirectiveStack.push_back(Loc);
94}
95
96void PPConditionalDirectiveRecord::Elif(SourceLocation Loc,
97 SourceRange ConditionRange,
98 SourceLocation IfLoc) {
99 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
100 CondDirectiveStack.back() = Loc;
101}
102
103void PPConditionalDirectiveRecord::Else(SourceLocation Loc,
104 SourceLocation IfLoc) {
105 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
106 CondDirectiveStack.back() = Loc;
107}
108
109void PPConditionalDirectiveRecord::Endif(SourceLocation Loc,
110 SourceLocation IfLoc) {
111 addCondDirectiveLoc(CondDirectiveLoc(Loc, CondDirectiveStack.back()));
112 assert(!CondDirectiveStack.empty());
113 CondDirectiveStack.pop_back();
114}
115
116size_t PPConditionalDirectiveRecord::getTotalMemory() const {
117 return llvm::capacity_in_bytes(CondDirectiveLocs);
118}