lib/treetop/compiler/node_classes/repetition.rb



module Treetop
  module Compiler
    class Repetition < ParsingExpression
      def compile(address, builder, parent_expression)
        super
        repeated_expression = parent_expression.atomic
        begin_comment(parent_expression)
        use_vars :result, :accumulator, :start_index

        builder.loop do
          obtain_new_subexpression_address
          repeated_expression.compile(subexpression_address, builder)
          builder.if__ subexpression_success? do
            accumulate_subexpression_result
          end
          builder.else_ do
            builder.break
          end
          if max && !max.empty?
            builder.if_ "#{accumulator_var}.size == #{max.text_value}" do
              builder.break
            end
          end
        end
      end

      def inline_module_name
        parent_expression.inline_module_name
      end

      def assign_and_extend_result parent_expression
        assign_result "instantiate_node(#{node_class_name},input, #{start_index_var}...index, #{accumulator_var})"
        extend_result_with_inline_module parent_expression
      end
    end


    class ZeroOrMore < Repetition
      def compile(address, builder, parent_expression)
        super
        assign_and_extend_result parent_expression
        end_comment(parent_expression)
      end

      def max
        nil
      end
    end

    class OneOrMore < Repetition
      def compile(address, builder, parent_expression)
        super
        builder.if__ "#{accumulator_var}.empty?" do
          reset_index
          assign_failure start_index_var
        end
        builder.else_ do
          assign_and_extend_result parent_expression
        end
        end_comment(parent_expression)
      end

      def max
        nil
      end

      def expected
        parent_expression.atomic.expected && '"at least one "+'+parent_expression.atomic.expected
      end
    end

    class OccurrenceRange < Repetition
      def compile(address, builder, parent_expression)
        super

        if !min.empty? && min.text_value.to_i != 0
          # We got some, but fewer than we wanted. There'll be a failure reported already
          builder.if__ "#{accumulator_var}.size < #{min.text_value}" do
            reset_index
            assign_failure start_index_var
          end
          builder.else_ do
            clean_unsaturated
            assign_and_extend_result parent_expression
          end
        else
          clean_unsaturated
          assign_and_extend_result parent_expression
        end

        end_comment(parent_expression)
      end

      # remove the last terminal_failure if we merely failed to reach the maximum
      def clean_unsaturated
        if !max.empty? && max.text_value.to_i > 0
          builder.if_ "#{accumulator_var}.size < #{max.text_value}" do
            builder << '@terminal_failures.pop'  # Ignore the last failure.
          end
        end
      end

      def expected
        parent_expression.atomic.expected && "at least #{min.text_value} "+parent_expression.atomic.expected
      end
    end

  end
end