Tim Keith | 18cee3e | 2018-05-01 19:50:34 | [diff] [blame] | 1 | // Copyright (c) 2018, NVIDIA CORPORATION. All rights reserved. |
| 2 | // |
| 3 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | // you may not use this file except in compliance with the License. |
| 5 | // You may obtain a copy of the License at |
| 6 | // |
| 7 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | // |
| 9 | // Unless required by applicable law or agreed to in writing, software |
| 10 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | // See the License for the specific language governing permissions and |
| 13 | // limitations under the License. |
| 14 | |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 15 | #include "parsing.h" |
| 16 | #include "grammar.h" |
peter klausler | 2c06be3 | 2018-04-19 20:03:23 | [diff] [blame] | 17 | #include "instrumented-parser.h" |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 18 | #include "message.h" |
peter klausler | efed300 | 2018-06-14 16:35:38 | [diff] [blame] | 19 | #include "openmp-grammar.h" |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 20 | #include "preprocessor.h" |
| 21 | #include "prescan.h" |
| 22 | #include "provenance.h" |
| 23 | #include "source.h" |
| 24 | #include <sstream> |
| 25 | |
Tim Keith | 7f66c0a | 2018-05-02 20:48:12 | [diff] [blame] | 26 | namespace Fortran::parser { |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 27 | |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 28 | void Parsing::Prescan(const std::string &path, Options options) { |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 29 | options_ = options; |
| 30 | |
| 31 | std::stringstream fileError; |
peter klausler | 2169993 | 2018-04-06 17:34:59 | [diff] [blame] | 32 | const SourceFile *sourceFile; |
| 33 | if (path == "-") { |
| 34 | sourceFile = allSources_.ReadStandardInput(&fileError); |
| 35 | } else { |
| 36 | sourceFile = allSources_.Open(path, &fileError); |
| 37 | } |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 38 | if (sourceFile == nullptr) { |
| 39 | ProvenanceRange range{allSources_.AddCompilerInsertion(path)}; |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 40 | MessageFormattedText msg("%s"_err_en_US, fileError.str().data()); |
peter klausler | e6b1f1f | 2018-04-23 19:17:11 | [diff] [blame] | 41 | messages_.Put(Message{range, std::move(msg)}); |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 42 | return; |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 43 | } |
peter klausler | 424ec7b | 2018-03-20 17:59:07 | [diff] [blame] | 44 | if (sourceFile->bytes() == 0) { |
| 45 | ProvenanceRange range{allSources_.AddCompilerInsertion(path)}; |
peter klausler | e6b1f1f | 2018-04-23 19:17:11 | [diff] [blame] | 46 | messages_.Put(Message{range, "file is empty"_err_en_US}); |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 47 | return; |
peter klausler | 424ec7b | 2018-03-20 17:59:07 | [diff] [blame] | 48 | } |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 49 | |
peter klausler | 2169993 | 2018-04-06 17:34:59 | [diff] [blame] | 50 | // N.B. Be sure to not push the search directory paths until the primary |
| 51 | // source file has been opened. If foo.f is missing from the current |
| 52 | // working directory, we don't want to accidentally read another foo.f |
| 53 | // from another directory that's on the search path. |
peter klausler | 0983fbc | 2018-03-16 23:13:49 | [diff] [blame] | 54 | for (const auto &path : options.searchDirectories) { |
| 55 | allSources_.PushSearchPathDirectory(path); |
| 56 | } |
| 57 | |
peter klausler | 75b2933 | 2018-03-23 22:14:52 | [diff] [blame] | 58 | Preprocessor preprocessor{allSources_}; |
peter klausler | 0983fbc | 2018-03-16 23:13:49 | [diff] [blame] | 59 | for (const auto &predef : options.predefinitions) { |
| 60 | if (predef.second.has_value()) { |
| 61 | preprocessor.Define(predef.first, *predef.second); |
| 62 | } else { |
| 63 | preprocessor.Undefine(predef.first); |
| 64 | } |
| 65 | } |
peter klausler | 75b2933 | 2018-03-23 22:14:52 | [diff] [blame] | 66 | Prescanner prescanner{messages_, cooked_, preprocessor}; |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 67 | prescanner.set_fixedForm(options.isFixedForm) |
| 68 | .set_fixedFormColumnLimit(options.fixedFormColumns) |
| 69 | .set_encoding(options.encoding) |
peter klausler | f701b42 | 2018-07-17 23:58:21 | [diff] [blame^] | 70 | .set_enableBackslashEscapesInCharLiterals( // TODO pmk |
| 71 | options.enabled.test(LanguageFeature::BackslashEscapes)) |
| 72 | .set_enableOldDebugLines( |
| 73 | options.enabled.test(LanguageFeature::OldDebugLines)) |
peter klausler | d5a084c | 2018-03-30 23:21:12 | [diff] [blame] | 74 | .set_warnOnNonstandardUsage(options_.isStrictlyStandard) |
peter klausler | 75b2933 | 2018-03-23 22:14:52 | [diff] [blame] | 75 | .AddCompilerDirectiveSentinel("dir$"); |
peter klausler | f701b42 | 2018-07-17 23:58:21 | [diff] [blame^] | 76 | if (options.enabled.test(LanguageFeature::OpenMP)) { |
Hongyon Suauthai | 6f7b4fa | 2018-05-16 14:28:54 | [diff] [blame] | 77 | prescanner.AddCompilerDirectiveSentinel("$omp"); |
| 78 | } |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 79 | ProvenanceRange range{ |
| 80 | allSources_.AddIncludedFile(*sourceFile, ProvenanceRange{})}; |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 81 | prescanner.Prescan(range); |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 82 | cooked_.Marshal(); |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 83 | } |
| 84 | |
| 85 | void Parsing::DumpCookedChars(std::ostream &out) const { |
peter klausler | c387b6d | 2018-04-20 18:23:51 | [diff] [blame] | 86 | UserState userState{cooked_}; |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 87 | ParseState parseState{cooked_}; |
| 88 | parseState.set_inFixedForm(options_.isFixedForm).set_userState(&userState); |
peter klausler | 10907c7 | 2018-04-02 23:33:10 | [diff] [blame] | 89 | while (std::optional<const char *> p{parseState.GetNextChar()}) { |
| 90 | out << **p; |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 91 | } |
| 92 | } |
| 93 | |
| 94 | void Parsing::DumpProvenance(std::ostream &out) const { cooked_.Dump(out); } |
| 95 | |
peter klausler | 56b09e0 | 2018-04-19 22:46:02 | [diff] [blame] | 96 | void Parsing::DumpParsingLog(std::ostream &out) const { |
| 97 | log_.Dump(out, cooked_); |
| 98 | } |
peter klausler | 2c06be3 | 2018-04-19 20:03:23 | [diff] [blame] | 99 | |
peter klausler | 30f337a | 2018-04-20 23:14:57 | [diff] [blame] | 100 | void Parsing::Parse(std::ostream *out) { |
peter klausler | c387b6d | 2018-04-20 18:23:51 | [diff] [blame] | 101 | UserState userState{cooked_}; |
peter klausler | 30f337a | 2018-04-20 23:14:57 | [diff] [blame] | 102 | userState.set_debugOutput(out) |
| 103 | .set_instrumentedParse(options_.instrumentedParse) |
| 104 | .set_log(&log_); |
peter klausler | f701b42 | 2018-07-17 23:58:21 | [diff] [blame^] | 105 | userState.Enable(options_.enabled); |
| 106 | userState.EnableWarnings(options_.warning); |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 107 | ParseState parseState{cooked_}; |
| 108 | parseState.set_inFixedForm(options_.isFixedForm) |
| 109 | .set_encoding(options_.encoding) |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 110 | .set_userState(&userState); |
peter klausler | c387b6d | 2018-04-20 18:23:51 | [diff] [blame] | 111 | parseTree_ = program.Parse(parseState); |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 112 | CHECK( |
peter klausler | f595462 | 2018-04-13 23:00:03 | [diff] [blame] | 113 | !parseState.anyErrorRecovery() || parseState.messages().AnyFatalError()); |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 114 | consumedWholeFile_ = parseState.IsAtEnd(); |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 115 | messages_.Annex(parseState.messages()); |
peter klausler | 43c64d2 | 2018-04-17 22:47:51 | [diff] [blame] | 116 | finalRestingPlace_ = parseState.GetLocation(); |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 117 | } |
| 118 | |
peter klausler | 7335482 | 2018-04-20 00:02:12 | [diff] [blame] | 119 | void Parsing::ClearLog() { log_.clear(); } |
| 120 | |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 121 | bool Parsing::ForTesting(std::string path, std::ostream &err) { |
| 122 | Prescan(path, Options{}); |
| 123 | if (messages_.AnyFatalError()) { |
peter klausler | 56b09e0 | 2018-04-19 22:46:02 | [diff] [blame] | 124 | messages_.Emit(err, cooked_); |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 125 | err << "could not scan " << path << '\n'; |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 126 | return false; |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 127 | } |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 128 | Parse(); |
peter klausler | 56b09e0 | 2018-04-19 22:46:02 | [diff] [blame] | 129 | messages_.Emit(err, cooked_); |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 130 | if (!consumedWholeFile_) { |
peter klausler | 5e69a75 | 2018-05-02 19:07:49 | [diff] [blame] | 131 | EmitMessage(err, finalRestingPlace_, "parser FAIL; final position"); |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 132 | return false; |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 133 | } |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 134 | if (messages_.AnyFatalError() || !parseTree_.has_value()) { |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 135 | err << "could not parse " << path << '\n'; |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 136 | return false; |
peter klausler | af22e07f | 2018-04-02 22:51:04 | [diff] [blame] | 137 | } |
peter klausler | 1e1ea72 | 2018-04-02 23:06:18 | [diff] [blame] | 138 | return true; |
peter klausler | 79d044e | 2018-03-01 00:56:10 | [diff] [blame] | 139 | } |
Tim Keith | 7f66c0a | 2018-05-02 20:48:12 | [diff] [blame] | 140 | |
| 141 | } // namespace Fortran::parser |