File: SwitchToIf.cc

package info (click to toggle)
rumur 2026.03.11-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 3,664 kB
  • sloc: cpp: 18,723; ansic: 3,824; python: 1,578; objc: 1,542; yacc: 568; sh: 331; lex: 241; lisp: 15; makefile: 5
file content (127 lines) | stat: -rw-r--r-- 3,147 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
#include "SwitchToIf.h"
#include "Stage.h"
#include <cassert>
#include <cstddef>
#include <ctype.h>
#include <rumur/rumur.h>
#include <string>

using namespace rumur;

SwitchToIf::SwitchToIf(Stage &next_) : IntermediateStage(next_) {}

void SwitchToIf::process(const Token &t) {

  // ignore any shift messages
  if (t.type == Token::SUBJ) {
    next.process(t);
    return;
  }

  // if we already understand indentation, we can just pass through
  if (learned_indentation) {
    next.process(t);
    return;
  }

  // if this character is a newline, assume we have an opportunity to learn
  // indentation coming up
  if (t.character == "\n") {
    last_newline = true;
    indentation = "";
    next.process(t);
    return;
  }

  // if we have not just seen a newline, assume we are not currently writing
  // indentation
  if (!last_newline) {
    next.process(t);
    return;
  }

  // if this is white space, continue accruing
  if (t.character.size() == 1 && isspace(t.character[0])) {
    indentation += t.character;

    // otherwise, if we have found a non-zero offset, we know the indentation
  } else if (indentation != "") {
    learned_indentation = true;

    // otherwise, we are looking at a non-indented line
  } else {
    last_newline = false;
  }

  next.process(t);
}

void SwitchToIf::visit_switch(const Switch &n) {

  // we need to emit the switched-on expression multiple times, so we require it
  // to be side-effect free
  if (!n.expr->is_pure())
    throw Error("impure expression in switch statement expression prevents "
                "transforming this into an if-then-else",
                n.loc);

  // write everything up to the start of this expression
  top->sync_to(n);

  // handle an edge case where the switch has no cases and should be omitted
  // entirely
  if (n.cases.empty()) {
    top->skip_to(n.loc.end);
    return;
  }

  // see if we can figure out what level of indentation this switch statement is
  // at
  assert(n.loc.begin.column > 0);
  size_t spaces = static_cast<size_t>(n.loc.begin.column) - 1;
  std::string indent;
  if (learned_indentation && spaces % indentation.size() == 0) {
    for (size_t i = 0; i * indentation.size() < spaces; i++)
      indent += indentation;
  }

  std::string sep;
  for (const SwitchCase &c : n.cases) {

    // seek up to this case
    top->skip_to(c);

    if (c.matches.empty()) {
      *top << sep << "e\n";
    } else {
      *top << sep << "if ";

      std::string sep2;
      bool brackets = c.matches.size() > 1;
      for (const Ptr<Expr> &m : c.matches) {
        *top << sep2 << (brackets ? "(" : "") << n.expr->to_string() << " = ";
        top->skip_to(*m);
        top->dispatch(*m);
        *top << (brackets ? ")" : "");
        sep2 = " | ";
      }

      *top << " then\n";
    }

    if (!c.body.empty()) {
      top->skip_to(*c.body[0]);
      if (learned_indentation)
        *top << indent << indentation;
    }
    for (const Ptr<Stmt> &s : c.body)
      top->dispatch(*s);
    if (!c.body.empty())
      *top << ";";

    sep = "\n" + indent + "els";
  }

  *top << "\n" << indent << "endif";
  top->skip_to(n.loc.end);
}