From 95fc7dd03ac49ddf1d155f27c64c6efd2934f250 Mon Sep 17 00:00:00 2001 From: AnotherTest Date: Sun, 12 Jul 2020 01:41:24 +0430 Subject: Shell: Parse lists serially, and flatten them only when needed This allows `((1 2 3) (4 5 6))` to remain nested until we explicitly flatten it out. --- Shell/AST.cpp | 99 ++++++++++++++++++++++++++++++++++++++++---------------- Shell/AST.h | 10 ++++-- Shell/Parser.cpp | 20 ++++++------ Shell/Shell.cpp | 12 +++---- 4 files changed, 94 insertions(+), 47 deletions(-) (limited to 'Shell') diff --git a/Shell/AST.cpp b/Shell/AST.cpp index 1e2261ba2e..fe167773c4 100644 --- a/Shell/AST.cpp +++ b/Shell/AST.cpp @@ -216,33 +216,59 @@ And::~And() void ListConcatenate::dump(int level) const { Node::dump(level); - m_element->dump(level + 1); - m_list->dump(level + 1); + for (auto& element : m_list) + element->dump(level + 1); } RefPtr ListConcatenate::run(RefPtr shell) { - auto list = m_list->run(shell)->resolve_without_cast(shell); - auto element = m_element->run(shell)->resolve_without_cast(shell); + RefPtr result = nullptr; - if (list->is_command() || element->is_command()) { - auto joined_commands = join_commands(element->resolve_as_commands(shell), list->resolve_as_commands(shell)); + for (auto& element : m_list) { + if (!result) { + result = create({ element->run(shell)->resolve_without_cast(shell) }); + continue; + } + auto element_value = element->run(shell)->resolve_without_cast(shell); + + if (result->is_command() || element_value->is_command()) { + auto joined_commands = join_commands(result->resolve_as_commands(shell), element_value->resolve_as_commands(shell)); + + if (joined_commands.size() == 1) + result = create(joined_commands[0]); + else + result = create(move(joined_commands)); + } else { + Vector> values; - if (joined_commands.size() == 1) - return create(joined_commands[0]); - return create(move(joined_commands)); + if (result->is_list_without_resolution()) { + values.append(static_cast(result.ptr())->values()); + } else { + for (auto& result : result->resolve_as_list(shell)) + values.append(create(result)); + } + + values.append(move(element_value)); + + result = create(move(values)); + } } + if (!result) + return create({}); - return create({ move(element), move(list) }); + return result; } void ListConcatenate::highlight_in_editor(Line::Editor& editor, Shell& shell, HighlightMetadata metadata) { auto first = metadata.is_first_in_list; metadata.is_first_in_list = false; - m_list->highlight_in_editor(editor, shell, metadata); + metadata.is_first_in_list = first; - m_element->highlight_in_editor(editor, shell, metadata); + for (auto& element : m_list) { + element->highlight_in_editor(editor, shell, metadata); + metadata.is_first_in_list = false; + } } HitTestResult ListConcatenate::hit_test_position(size_t offset) @@ -250,29 +276,37 @@ HitTestResult ListConcatenate::hit_test_position(size_t offset) if (!position().contains(offset)) return {}; - auto result = m_element->hit_test_position(offset); - if (result.matching_node) - return result; - result = m_list->hit_test_position(offset); - if (!result.closest_node_with_semantic_meaning) - result.closest_node_with_semantic_meaning = this; - return result; + bool first = true; + for (auto& element : m_list) { + auto result = element->hit_test_position(offset); + if (!result.closest_node_with_semantic_meaning && !first) + result.closest_node_with_semantic_meaning = this; + if (result.matching_node) + return result; + first = false; + } + + return {}; } RefPtr ListConcatenate::leftmost_trivial_literal() const { - return m_element->leftmost_trivial_literal(); + if (m_list.is_empty()) + return nullptr; + + return m_list.first()->leftmost_trivial_literal(); } -ListConcatenate::ListConcatenate(Position position, RefPtr element, RefPtr list) +ListConcatenate::ListConcatenate(Position position, Vector> list) : Node(move(position)) - , m_element(move(element)) , m_list(move(list)) { - if (m_element->is_syntax_error()) - set_is_syntax_error(m_element->syntax_error_node()); - else if (m_list->is_syntax_error()) - set_is_syntax_error(m_list->syntax_error_node()); + for (auto& element : m_list) { + if (element->is_syntax_error()) { + set_is_syntax_error(element->syntax_error_node()); + break; + } + } } ListConcatenate::~ListConcatenate() @@ -451,9 +485,9 @@ RefPtr CastToList::run(RefPtr shell) if (!m_inner) return create({}); - auto inner_value = m_inner->run(shell); + auto inner_value = m_inner->run(shell)->resolve_without_cast(shell); - if (inner_value->is_command()) + if (inner_value->is_command() || inner_value->is_list()) return inner_value; auto values = inner_value->resolve_as_list(shell); @@ -1783,6 +1817,15 @@ Vector ListValue::resolve_as_list(RefPtr shell) return values; } +RefPtr ListValue::resolve_without_cast(RefPtr shell) +{ + Vector> values; + for (auto& value : m_contained_values) + values.append(value->resolve_without_cast(shell)); + + return create(move(values)); +} + CommandValue::~CommandValue() { } diff --git a/Shell/AST.h b/Shell/AST.h index 2e9c74ea5c..58945126f4 100644 --- a/Shell/AST.h +++ b/Shell/AST.h @@ -164,6 +164,7 @@ public: virtual bool is_job() const { return false; } virtual bool is_list() const { return false; } virtual bool is_string() const { return false; } + virtual bool is_list_without_resolution() const { return false; } }; class CommandValue final : public Value { @@ -221,14 +222,18 @@ private: class ListValue final : public Value { public: virtual Vector resolve_as_list(RefPtr) override; + virtual RefPtr resolve_without_cast(RefPtr) override; virtual ~ListValue(); virtual bool is_list() const override { return true; } + virtual bool is_list_without_resolution() const override { return true; } ListValue(Vector values); ListValue(Vector> values) : m_contained_values(move(values)) { } + const Vector>& values() const { return m_contained_values; } + private: Vector> m_contained_values; }; @@ -389,7 +394,7 @@ private: class ListConcatenate final : public Node { public: - ListConcatenate(Position, RefPtr, RefPtr); + ListConcatenate(Position, Vector>); virtual ~ListConcatenate(); private: @@ -401,8 +406,7 @@ private: virtual bool is_list() const override { return true; } virtual RefPtr leftmost_trivial_literal() const override; - RefPtr m_element; - RefPtr m_list; + Vector> m_list; }; class Background final : public Node { diff --git a/Shell/Parser.cpp b/Shell/Parser.cpp index c19f2d3664..4a56761f65 100644 --- a/Shell/Parser.cpp +++ b/Shell/Parser.cpp @@ -420,19 +420,19 @@ RefPtr Parser::parse_list_expression() consume_while(is_whitespace); auto rule_start = push_start(); + Vector> nodes; - auto expr = parse_expression(); - if (!expr) - return nullptr; + do { + auto expr = parse_expression(); + if (!expr) + break; + nodes.append(move(expr)); + } while (!consume_while(is_whitespace).is_empty()); - if (consume_while(is_whitespace).is_empty()) - return expr; - - auto list = parse_list_expression(); - if (!list) - return create(move(expr)); + if (nodes.is_empty()) + return nullptr; - return create(move(expr), move(list)); // Join Element List + return create(move(nodes)); // Concatenate List } RefPtr Parser::parse_expression() diff --git a/Shell/Shell.cpp b/Shell/Shell.cpp index 4909b3e8b9..a7cfbcaaae 100644 --- a/Shell/Shell.cpp +++ b/Shell/Shell.cpp @@ -361,19 +361,19 @@ int Shell::run_command(const StringView& cmd) if (!command) return 0; +#ifdef SH_DEBUG + dbg() << "Command follows"; + command->dump(0); +#endif + if (command->is_syntax_error()) { auto& error_node = command->syntax_error_node(); auto& position = error_node.position(); fprintf(stderr, "Shell: Syntax error in command: %s\n", error_node.error_text().characters()); - fprintf(stderr, "Around '%.*s'\n", (int)min(position.end_offset - position.start_offset, (size_t)10), cmd.characters_without_null_termination() + position.start_offset); + fprintf(stderr, "Around '%.*s' at %zu:%zu\n", (int)min(position.end_offset - position.start_offset, (size_t)10), cmd.characters_without_null_termination() + position.start_offset, position.start_offset, position.end_offset); return 1; } -#ifdef SH_DEBUG - dbg() << "Command follows"; - command->dump(0); -#endif - tcgetattr(0, &termios); auto result = command->run(*this); -- cgit v1.2.3