summaryrefslogtreecommitdiff
path: root/frontend/driver.cc
blob: 1c20d097c2d5a092d5671366c8bf3da2c9717fee (plain)
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
/* Parsing driver.
   Copyright (C) 2025 Free Software Foundation, Inc.

GCC is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.

GCC is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with GCC; see the file COPYING3.  If not see
<http://www.gnu.org/licenses/>.  */

#include "elna/frontend/driver.h"

namespace elna::frontend
{
    position make_position(const yy::location& location)
    {
        position result;
        result.line = static_cast<std::size_t>(location.begin.line);
        result.column = static_cast<std::size_t>(location.begin.column);

        return result;
    }

    syntax_error::syntax_error(const std::string& message,
            const char *input_file, const yy::location& location)
        : error(input_file, make_position(location)), message(message)
    {
    }

    std::string syntax_error::what() const
    {
        return message;
    }

    driver::driver(const char *input_file)
        : error_container(input_file)
    {
    }

    char escape_char(char escape)
    {
        switch (escape)
        {
        case 'n':
            return '\n';
        case 'a':
            return '\a';
        case 'b':
            return '\b';
        case 't':
            return '\t';
        case 'f':
            return '\f';
        case 'r':
            return '\r';
        case 'v':
            return '\v';
        case '\\':
            return '\\';
        case '\'':
            return '\'';
        case '"':
            return '"';
        case '?':
            return '\?';
        case '0':
            return '\0';
        default:
            return escape_invalid_char;
        }
    }

    std::optional<std::string> escape_string(const char *escape)
    {
        std::string result;
        const char *current_position = escape + 1;

        while (*current_position != '\0')
        {
            if (*current_position == '\\' && *(current_position + 1) == 'x')
            {
                current_position += 2;

                std::size_t processed;
                char character = static_cast<char>(std::stoi(current_position, &processed, 16));
                if (processed == 0)
                {
                    return std::nullopt;
                }
                else
                {
                    current_position += processed - 1;
                    result.push_back(character);
                }
            }
            else if (*current_position == '\\')
            {
                ++current_position;

                char escape = escape_char(*current_position);
                if (escape == escape_invalid_char)
                {
                    return std::nullopt;
                }
                result.push_back(escape);
            }
            else
            {
                result.push_back(*current_position);
            }
            ++current_position;
        }
        result.pop_back(); // Remove the terminating quote character.

        return result;
    }
}