mirror of
https://github.com/RPCS3/llvm.git
synced 2025-05-15 09:56:02 +00:00

When preprocessing resource scripts (which can easily be done outside of llvm-rc), included headers can leave behind C declarations (despite preprocessing with -DRC_INVOKED), that can't be parsed by a resource compiler. This is handled in all of rc.exe, by parsing the preprocessor output line markers and ignoring content from files named *.h and *.c, documented at [1]. In addition to this filtering, strip out any other preprocessor directive that is left behind (like pragmas) which also can't be handled by the tokenizer. The added test uses both standard #line markers (supported by rc.exe) and GNU style extended line markers, thus this test doesn't pass with rc.exe, but passes with GNU windres. (Windres on the other hand doesn't filter out files named *.c, only *.h.) Differential Revision: https://reviews.llvm.org/D46579 [1] https://msdn.microsoft.com/en-us/library/windows/desktop/aa381033(v=vs.85).aspx git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@331903 91177308-0d34-0410-b5e6-96231b3b80d8
113 lines
2.7 KiB
C++
113 lines
2.7 KiB
C++
//===-- ResourceScriptCppFilter.cpp ----------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===---------------------------------------------------------------------===//
|
|
//
|
|
// This file implements an interface defined in ResourceScriptCppFilter.h.
|
|
//
|
|
//===---------------------------------------------------------------------===//
|
|
|
|
#include "ResourceScriptCppFilter.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
|
|
#include <vector>
|
|
|
|
using namespace llvm;
|
|
|
|
namespace {
|
|
|
|
class Filter {
|
|
public:
|
|
explicit Filter(StringRef Input) : Data(Input), DataLength(Input.size()) {}
|
|
|
|
std::string run();
|
|
|
|
private:
|
|
// Parse the line, returning whether the line should be included in
|
|
// the output.
|
|
bool parseLine(StringRef Line);
|
|
|
|
bool streamEof() const;
|
|
|
|
StringRef Data;
|
|
size_t DataLength;
|
|
|
|
size_t Pos = 0;
|
|
bool Outputting = true;
|
|
};
|
|
|
|
std::string Filter::run() {
|
|
std::vector<StringRef> Output;
|
|
|
|
while (!streamEof() && Pos != StringRef::npos) {
|
|
size_t LineStart = Pos;
|
|
Pos = Data.find_first_of("\r\n", Pos);
|
|
Pos = Data.find_first_not_of("\r\n", Pos);
|
|
StringRef Line = Data.take_front(Pos).drop_front(LineStart);
|
|
|
|
if (parseLine(Line))
|
|
Output.push_back(Line);
|
|
}
|
|
|
|
return llvm::join(Output, "");
|
|
}
|
|
|
|
bool Filter::parseLine(StringRef Line) {
|
|
Line = Line.ltrim();
|
|
|
|
if (!Line.consume_front("#")) {
|
|
// A normal content line, filtered according to the current mode.
|
|
return Outputting;
|
|
}
|
|
|
|
// Found a preprocessing directive line. From here on, we always return
|
|
// false since the preprocessing directives should be filtered out.
|
|
|
|
Line.consume_front("line");
|
|
if (!Line.startswith(" "))
|
|
return false; // Not a line directive (pragma etc).
|
|
|
|
// #line 123 "path/file.h"
|
|
// # 123 "path/file.h" 1
|
|
|
|
Line =
|
|
Line.ltrim(); // There could be multiple spaces after the #line directive
|
|
|
|
size_t N;
|
|
if (Line.consumeInteger(10, N)) // Returns true to signify an error
|
|
return false;
|
|
|
|
Line = Line.ltrim();
|
|
|
|
if (!Line.consume_front("\""))
|
|
return false; // Malformed line, no quote found.
|
|
|
|
// Split the string at the last quote (in case the path name had
|
|
// escaped quotes as well).
|
|
Line = Line.rsplit('"').first;
|
|
|
|
StringRef Ext = Line.rsplit('.').second;
|
|
|
|
if (Ext.equals_lower("h") || Ext.equals_lower("c")) {
|
|
Outputting = false;
|
|
} else {
|
|
Outputting = true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool Filter::streamEof() const { return Pos == DataLength; }
|
|
|
|
} // anonymous namespace
|
|
|
|
namespace llvm {
|
|
|
|
std::string filterCppOutput(StringRef Input) { return Filter(Input).run(); }
|
|
|
|
} // namespace llvm
|