Chris@16
|
1 /*=============================================================================
|
Chris@16
|
2 Boost.Wave: A Standard compliant C++ preprocessor library
|
Chris@16
|
3
|
Chris@16
|
4 http://www.boost.org/
|
Chris@16
|
5
|
Chris@16
|
6 Copyright (c) 2001-2012 Hartmut Kaiser. Distributed under the Boost
|
Chris@16
|
7 Software License, Version 1.0. (See accompanying file
|
Chris@16
|
8 LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
9 =============================================================================*/
|
Chris@16
|
10
|
Chris@16
|
11 #if !defined(CPP_IFBLOCK_HPP_D4676B36_00C5_41F4_BC9F_9CBBAE3B8006_INCLUDED)
|
Chris@16
|
12 #define CPP_IFBLOCK_HPP_D4676B36_00C5_41F4_BC9F_9CBBAE3B8006_INCLUDED
|
Chris@16
|
13
|
Chris@16
|
14 #include <stack>
|
Chris@16
|
15 #include <boost/wave/wave_config.hpp>
|
Chris@16
|
16
|
Chris@16
|
17 // this must occur after all of the includes and before any code appears
|
Chris@16
|
18 #ifdef BOOST_HAS_ABI_HEADERS
|
Chris@16
|
19 #include BOOST_ABI_PREFIX
|
Chris@16
|
20 #endif
|
Chris@16
|
21
|
Chris@16
|
22 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
23 namespace boost {
|
Chris@16
|
24 namespace wave {
|
Chris@16
|
25 namespace util {
|
Chris@16
|
26
|
Chris@16
|
27 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
28 // the class if_blocks handles recursive conditional compilation contexts
|
Chris@16
|
29 class if_block
|
Chris@16
|
30 {
|
Chris@16
|
31 public:
|
Chris@16
|
32 if_block() :
|
Chris@16
|
33 status(true), some_part_status(true),
|
Chris@16
|
34 enclosing_status(true), is_in_else(false)
|
Chris@16
|
35 {
|
Chris@16
|
36 }
|
Chris@16
|
37 if_block(bool status_, bool enclosing_status_) :
|
Chris@16
|
38 status(status_),
|
Chris@16
|
39 some_part_status(status_),
|
Chris@16
|
40 enclosing_status(enclosing_status_),
|
Chris@16
|
41 is_in_else(false)
|
Chris@16
|
42 {
|
Chris@16
|
43 }
|
Chris@16
|
44
|
Chris@16
|
45 void set_status(bool status_)
|
Chris@16
|
46 {
|
Chris@16
|
47 status = status_;
|
Chris@16
|
48 if (status_)
|
Chris@16
|
49 some_part_status = true;
|
Chris@16
|
50 }
|
Chris@16
|
51 bool get_status() const { return status; }
|
Chris@16
|
52 bool get_some_part_status() const { return some_part_status; }
|
Chris@16
|
53 bool get_enclosing_status() const { return enclosing_status; }
|
Chris@16
|
54 bool get_in_else() const { return is_in_else; }
|
Chris@16
|
55 void set_in_else() { is_in_else = true; }
|
Chris@16
|
56
|
Chris@16
|
57 private:
|
Chris@16
|
58 bool status; // Current block is true
|
Chris@16
|
59 bool some_part_status; // One of the preceding or current #if/#elif was true
|
Chris@16
|
60 bool enclosing_status; // Enclosing #if block is true
|
Chris@16
|
61 bool is_in_else; // Inside the #else part
|
Chris@16
|
62 };
|
Chris@16
|
63
|
Chris@16
|
64 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
65 // stack of conditional compilation contexts
|
Chris@16
|
66 class if_block_stack
|
Chris@16
|
67 : private std::stack<if_block>
|
Chris@16
|
68 {
|
Chris@16
|
69 public:
|
Chris@16
|
70 typedef std::stack<if_block>::size_type size_type;
|
Chris@16
|
71
|
Chris@16
|
72 void enter_if_block(bool new_status)
|
Chris@16
|
73 {
|
Chris@16
|
74 // If enclosing block is false, then this block is also false
|
Chris@16
|
75 bool enclosing_status = get_status();
|
Chris@16
|
76 this->push (value_type (new_status && enclosing_status, enclosing_status));
|
Chris@16
|
77 }
|
Chris@16
|
78 bool enter_elif_block(bool new_status)
|
Chris@16
|
79 {
|
Chris@16
|
80 if (!is_inside_ifpart())
|
Chris@16
|
81 return false; // #elif without matching #if
|
Chris@16
|
82
|
Chris@16
|
83 if (get_enclosing_status()) {
|
Chris@16
|
84 if (get_status()) {
|
Chris@16
|
85 // entered a (false) #elif block from a true block
|
Chris@16
|
86 this->top().set_status(false);
|
Chris@16
|
87 }
|
Chris@16
|
88 else if (new_status && !this->top().get_some_part_status()) {
|
Chris@16
|
89 // Entered true #elif block and no previous block was true
|
Chris@16
|
90 this->top().set_status(new_status);
|
Chris@16
|
91 }
|
Chris@16
|
92 }
|
Chris@16
|
93 return true;
|
Chris@16
|
94 }
|
Chris@16
|
95 bool enter_else_block()
|
Chris@16
|
96 {
|
Chris@16
|
97 if (!is_inside_ifpart())
|
Chris@16
|
98 return false; // #else without matching #if
|
Chris@16
|
99
|
Chris@16
|
100 if (get_enclosing_status()) {
|
Chris@16
|
101 if (!this->top().get_some_part_status()) {
|
Chris@16
|
102 // Entered (true) #else block and no previous block was true
|
Chris@16
|
103 this->top().set_status(true);
|
Chris@16
|
104 }
|
Chris@16
|
105 else if (get_status()) {
|
Chris@16
|
106 // Entered (false) #else block from true block
|
Chris@16
|
107 this->top().set_status(false);
|
Chris@16
|
108 }
|
Chris@16
|
109
|
Chris@16
|
110 // Set else flag
|
Chris@16
|
111 this->top().set_in_else();
|
Chris@16
|
112 }
|
Chris@16
|
113 return true;
|
Chris@16
|
114 }
|
Chris@16
|
115 bool exit_if_block()
|
Chris@16
|
116 {
|
Chris@16
|
117 if (0 == this->size())
|
Chris@16
|
118 return false; // #endif without matching #if
|
Chris@16
|
119
|
Chris@16
|
120 this->pop();
|
Chris@16
|
121 return true;
|
Chris@16
|
122 }
|
Chris@16
|
123
|
Chris@16
|
124 // return, whether the top (innermost) condition is true or false
|
Chris@16
|
125 bool get_status() const
|
Chris@16
|
126 {
|
Chris@16
|
127 return 0 == this->size() || this->top().get_status();
|
Chris@16
|
128 }
|
Chris@16
|
129 bool get_some_part_status() const
|
Chris@16
|
130 {
|
Chris@16
|
131 return 0 == this->size() || this->top().get_some_part_status();
|
Chris@16
|
132 }
|
Chris@16
|
133 bool get_enclosing_status() const
|
Chris@16
|
134 {
|
Chris@16
|
135 return 0 == this->size() || this->top().get_enclosing_status();
|
Chris@16
|
136 }
|
Chris@16
|
137
|
Chris@16
|
138 size_type get_if_block_depth() const { return this->size(); }
|
Chris@16
|
139
|
Chris@16
|
140 protected:
|
Chris@16
|
141 bool is_inside_ifpart() const
|
Chris@16
|
142 {
|
Chris@16
|
143 return 0 != this->size() && !this->top().get_in_else();
|
Chris@16
|
144 }
|
Chris@16
|
145 bool is_inside_elsepart() const
|
Chris@16
|
146 {
|
Chris@16
|
147 return 0 != this->size() && this->top().get_in_else();
|
Chris@16
|
148 }
|
Chris@16
|
149 };
|
Chris@16
|
150
|
Chris@16
|
151 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
152 } // namespace util
|
Chris@16
|
153 } // namespace wave
|
Chris@16
|
154 } // namespace boost
|
Chris@16
|
155
|
Chris@16
|
156 // the suffix header occurs after all of the code
|
Chris@16
|
157 #ifdef BOOST_HAS_ABI_HEADERS
|
Chris@16
|
158 #include BOOST_ABI_SUFFIX
|
Chris@16
|
159 #endif
|
Chris@16
|
160
|
Chris@16
|
161 #endif // !defined(CPP_IFBLOCK_HPP_D4676B36_00C5_41F4_BC9F_9CBBAE3B8006_INCLUDED)
|