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 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
|
/* parse.c
Copyright (C) 2002, 2003, 2008 Niels Möller
This file is part of GNU Nettle.
GNU Nettle is free software: you can redistribute it and/or
modify it under the terms of either:
* the GNU Lesser General Public License as published by the Free
Software Foundation; either version 3 of the License, or (at your
option) any later version.
or
* the GNU General Public License as published by the Free
Software Foundation; either version 2 of the License, or (at your
option) any later version.
or both in parallel, as here.
GNU Nettle 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 copies of the GNU General Public License and
the GNU Lesser General Public License along with this program. If
not, see http://www.gnu.org/licenses/.
*/
#if HAVE_CONFIG_H
# include "config.h"
#endif
#include <assert.h>
#include <stdlib.h>
#include "parse.h"
#include "input.h"
void
sexp_compound_token_init(struct sexp_compound_token *token)
{
token->type = 0;
nettle_buffer_init(&token->display);
nettle_buffer_init(&token->string);
}
void
sexp_compound_token_clear(struct sexp_compound_token *token)
{
nettle_buffer_clear(&token->display);
nettle_buffer_clear(&token->string);
}
void
sexp_parse_init(struct sexp_parser *parser,
struct sexp_input *input,
enum sexp_mode mode)
{
parser->input = input;
parser->mode = mode;
/* Start counting with 1 for the top level, to make comparisons
* between transport and level simpler.
*
* FIXME: Is that trick ugly? */
parser->level = 1;
parser->transport = 0;
}
/* Get next token, and check that it is of the expected kind. */
static void
sexp_check_token(struct sexp_parser *parser,
enum sexp_token token,
struct nettle_buffer *string)
{
sexp_get_token(parser->input,
parser->transport ? SEXP_CANONICAL : parser->mode,
string);
if (parser->input->token != token)
die("Syntax error.\n");
}
/* Performs further processing of the input, in particular display
* types and transport decoding.
*
* This is complicated a little by the requirement that a
* transport-encoded block, {xxxxx}, must include exactly one
* expression. We check at the end of strings and list whether or not
* we should expect a SEXP_CODING_END as the next token. */
void
sexp_parse(struct sexp_parser *parser,
struct sexp_compound_token *token)
{
for (;;)
{
sexp_get_token(parser->input,
parser->transport ? SEXP_CANONICAL : parser->mode,
&token->string);
switch(parser->input->token)
{
case SEXP_LIST_END:
if (parser->level == parser->transport)
die("Unmatched end of list in transport encoded data.\n");
parser->level--;
if (!parser->level)
die("Unmatched end of list.\n");
token->type = SEXP_LIST_END;
check_transport_end:
if (parser->level == parser->transport)
{
sexp_check_token(parser, SEXP_CODING_END, &token->string);
assert(parser->transport);
assert(parser->level == parser->transport);
parser->level--;
parser->transport = 0;
}
return;
case SEXP_EOF:
if (parser->level > 1)
die("Unexpected end of file.\n");
token->type = SEXP_EOF;
return;
case SEXP_LIST_START:
parser->level++;
token->type = SEXP_LIST_START;
return;
case SEXP_DISPLAY_START:
sexp_check_token(parser, SEXP_STRING, &token->display);
sexp_check_token(parser, SEXP_DISPLAY_END, &token->display);
sexp_check_token(parser, SEXP_STRING, &token->string);
token->type = SEXP_DISPLAY;
goto check_transport_end;
case SEXP_STRING:
token->type = SEXP_STRING;
goto check_transport_end;
case SEXP_COMMENT:
token->type = SEXP_COMMENT;
return;
case SEXP_TRANSPORT_START:
if (parser->mode == SEXP_CANONICAL)
die("Base64 not allowed in canonical mode.\n");
parser->level++;
parser->transport = parser->level;
continue;
case SEXP_CODING_END:
die("Unexpected end of transport encoding.\n");
case SEXP_DISPLAY_END:
die("Unexpected end of display tag.\n");
case SEXP_DISPLAY:
/* Internal error. */
abort();
}
}
}
|