• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "Formatter.h"
18 
19 #include <assert.h>
20 
21 #include <android-base/logging.h>
22 
23 namespace android {
24 
Formatter()25 Formatter::Formatter() : mFile(nullptr /* invalid */), mIndentDepth(0), mAtStartOfLine(true) {}
26 
Formatter(FILE * file,size_t spacesPerIndent)27 Formatter::Formatter(FILE* file, size_t spacesPerIndent)
28     : mFile(file == nullptr ? stdout : file),
29       mIndentDepth(0),
30       mSpacesPerIndent(spacesPerIndent),
31       mAtStartOfLine(true) {}
32 
~Formatter()33 Formatter::~Formatter() {
34     if (mFile != stdout) {
35         fclose(mFile);
36     }
37     mFile = nullptr;
38 }
39 
indent(size_t level)40 void Formatter::indent(size_t level) {
41     mIndentDepth += level;
42 }
43 
unindent(size_t level)44 void Formatter::unindent(size_t level) {
45     assert(mIndentDepth >= level);
46     mIndentDepth -= level;
47 }
48 
indent(size_t level,const std::function<void (void)> & func)49 Formatter& Formatter::indent(size_t level, const std::function<void(void)>& func) {
50     this->indent(level);
51     func();
52     this->unindent(level);
53     return *this;
54 }
55 
indent(const std::function<void (void)> & func)56 Formatter& Formatter::indent(const std::function<void(void)>& func) {
57     return this->indent(1, func);
58 }
59 
block(const std::function<void (void)> & func)60 Formatter& Formatter::block(const std::function<void(void)>& func) {
61     (*this) << "{\n";
62     this->indent(func);
63     return (*this) << "}";
64 }
65 
setLinePrefix(const std::string & prefix)66 void Formatter::setLinePrefix(const std::string &prefix) {
67     mLinePrefix = prefix;
68 }
69 
unsetLinePrefix()70 void Formatter::unsetLinePrefix() {
71     mLinePrefix = "";
72 }
73 
endl()74 Formatter &Formatter::endl() {
75     return (*this) << "\n";
76 }
77 
sIf(const std::string & cond,const std::function<void (void)> & block)78 Formatter& Formatter::sIf(const std::string& cond, const std::function<void(void)>& block) {
79     (*this) << "if (" << cond << ") ";
80     return this->block(block);
81 }
82 
sElseIf(const std::string & cond,const std::function<void (void)> & block)83 Formatter& Formatter::sElseIf(const std::string& cond, const std::function<void(void)>& block) {
84     (*this) << " else if (" << cond << ") ";
85     return this->block(block);
86 }
87 
sElse(const std::function<void (void)> & block)88 Formatter& Formatter::sElse(const std::function<void(void)>& block) {
89     (*this) << " else ";
90     return this->block(block);
91 }
92 
sFor(const std::string & stmts,const std::function<void (void)> & block)93 Formatter& Formatter::sFor(const std::string& stmts, const std::function<void(void)>& block) {
94     (*this) << "for (" << stmts << ") ";
95     return this->block(block);
96 }
97 
sTry(const std::function<void (void)> & block)98 Formatter& Formatter::sTry(const std::function<void(void)>& block) {
99     (*this) << "try ";
100     return this->block(block);
101 }
102 
sCatch(const std::string & exception,const std::function<void (void)> & block)103 Formatter& Formatter::sCatch(const std::string& exception, const std::function<void(void)>& block) {
104     (*this) << " catch (" << exception << ") ";
105     return this->block(block);
106 }
107 
sFinally(const std::function<void (void)> & block)108 Formatter& Formatter::sFinally(const std::function<void(void)>& block) {
109     (*this) << " finally ";
110     return this->block(block);
111 }
112 
sWhile(const std::string & cond,const std::function<void (void)> & block)113 Formatter& Formatter::sWhile(const std::string& cond, const std::function<void(void)>& block) {
114     (*this) << "while (" << cond << ") ";
115     return this->block(block);
116 }
117 
operator <<(const std::string & out)118 Formatter &Formatter::operator<<(const std::string &out) {
119     const size_t len = out.length();
120     size_t start = 0;
121     while (start < len) {
122         size_t pos = out.find('\n', start);
123 
124         if (pos == std::string::npos) {
125             if (mAtStartOfLine) {
126                 fprintf(mFile, "%*s", (int)(mSpacesPerIndent * mIndentDepth), "");
127                 fprintf(mFile, "%s", mLinePrefix.c_str());
128                 mAtStartOfLine = false;
129             }
130 
131             output(out.substr(start));
132             break;
133         }
134 
135         if (mAtStartOfLine && (pos > start || !mLinePrefix.empty())) {
136             fprintf(mFile, "%*s", (int)(mSpacesPerIndent * mIndentDepth), "");
137             fprintf(mFile, "%s", mLinePrefix.c_str());
138         }
139 
140         if (pos == start) {
141             fprintf(mFile, "\n");
142             mAtStartOfLine = true;
143         } else if (pos > start) {
144             output(out.substr(start, pos - start + 1));
145             mAtStartOfLine = true;
146         }
147 
148         start = pos + 1;
149     }
150 
151     return *this;
152 }
153 
154 // NOLINT to suppress missing parentheses warning about __type__.
155 #define FORMATTER_INPUT_INTEGER(__type__)                       \
156     Formatter& Formatter::operator<<(__type__ n) { /* NOLINT */ \
157         return (*this) << std::to_string(n);                    \
158     }
159 
160 FORMATTER_INPUT_INTEGER(short);
161 FORMATTER_INPUT_INTEGER(unsigned short);
162 FORMATTER_INPUT_INTEGER(int);
163 FORMATTER_INPUT_INTEGER(unsigned int);
164 FORMATTER_INPUT_INTEGER(long);
165 FORMATTER_INPUT_INTEGER(unsigned long);
166 FORMATTER_INPUT_INTEGER(long long);
167 FORMATTER_INPUT_INTEGER(unsigned long long);
168 FORMATTER_INPUT_INTEGER(float);
169 FORMATTER_INPUT_INTEGER(double);
170 FORMATTER_INPUT_INTEGER(long double);
171 
172 #undef FORMATTER_INPUT_INTEGER
173 
174 // NOLINT to suppress missing parentheses warning about __type__.
175 #define FORMATTER_INPUT_CHAR(__type__)                          \
176     Formatter& Formatter::operator<<(__type__ c) { /* NOLINT */ \
177         return (*this) << std::string(1, (char)c);              \
178     }
179 
180 FORMATTER_INPUT_CHAR(char);
181 FORMATTER_INPUT_CHAR(signed char);
182 FORMATTER_INPUT_CHAR(unsigned char);
183 
184 #undef FORMATTER_INPUT_CHAR
185 
setNamespace(const std::string & space)186 void Formatter::setNamespace(const std::string &space) {
187     mSpace = space;
188 }
189 
isValid() const190 bool Formatter::isValid() const {
191     return mFile != nullptr;
192 }
193 
output(const std::string & text) const194 void Formatter::output(const std::string &text) const {
195     CHECK(isValid());
196 
197     fprintf(mFile, "%s", text.c_str());
198 }
199 
200 }  // namespace android
201