[yaml2obj/obj2yaml] - Add support for .stack_sizes sections.
[llvm-complete.git] / tools / llvm-rc / ResourceScriptCppFilter.cpp
blobe610be99dfb4d2476d105743cff62e2e2749a104
1 //===-- ResourceScriptCppFilter.cpp ----------------------------*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===---------------------------------------------------------------------===//
8 //
9 // This file implements an interface defined in ResourceScriptCppFilter.h.
11 //===---------------------------------------------------------------------===//
13 #include "ResourceScriptCppFilter.h"
14 #include "llvm/ADT/StringExtras.h"
16 #include <vector>
18 using namespace llvm;
20 namespace {
22 class Filter {
23 public:
24 explicit Filter(StringRef Input) : Data(Input), DataLength(Input.size()) {}
26 std::string run();
28 private:
29 // Parse the line, returning whether the line should be included in
30 // the output.
31 bool parseLine(StringRef Line);
33 bool streamEof() const;
35 StringRef Data;
36 size_t DataLength;
38 size_t Pos = 0;
39 bool Outputting = true;
42 std::string Filter::run() {
43 std::vector<StringRef> Output;
45 while (!streamEof() && Pos != StringRef::npos) {
46 size_t LineStart = Pos;
47 Pos = Data.find_first_of("\r\n", Pos);
48 Pos = Data.find_first_not_of("\r\n", Pos);
49 StringRef Line = Data.take_front(Pos).drop_front(LineStart);
51 if (parseLine(Line))
52 Output.push_back(Line);
55 return llvm::join(Output, "");
58 bool Filter::parseLine(StringRef Line) {
59 Line = Line.ltrim();
61 if (!Line.consume_front("#")) {
62 // A normal content line, filtered according to the current mode.
63 return Outputting;
66 // Found a preprocessing directive line. From here on, we always return
67 // false since the preprocessing directives should be filtered out.
69 Line.consume_front("line");
70 if (!Line.startswith(" "))
71 return false; // Not a line directive (pragma etc).
73 // #line 123 "path/file.h"
74 // # 123 "path/file.h" 1
76 Line =
77 Line.ltrim(); // There could be multiple spaces after the #line directive
79 size_t N;
80 if (Line.consumeInteger(10, N)) // Returns true to signify an error
81 return false;
83 Line = Line.ltrim();
85 if (!Line.consume_front("\""))
86 return false; // Malformed line, no quote found.
88 // Split the string at the last quote (in case the path name had
89 // escaped quotes as well).
90 Line = Line.rsplit('"').first;
92 StringRef Ext = Line.rsplit('.').second;
94 if (Ext.equals_lower("h") || Ext.equals_lower("c")) {
95 Outputting = false;
96 } else {
97 Outputting = true;
100 return false;
103 bool Filter::streamEof() const { return Pos == DataLength; }
105 } // anonymous namespace
107 namespace llvm {
109 std::string filterCppOutput(StringRef Input) { return Filter(Input).run(); }
111 } // namespace llvm