2026-01-27 19:56:33 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
|
|
|
|
|
2026-01-24 15:34:10 +01:00
|
|
|
#include "grammar_advanced.h"
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2026-01-24 16:13:16 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
2026-01-24 15:34:10 +01:00
|
|
|
|
|
|
|
|
#include "grammar_basic.h"
|
|
|
|
|
|
2026-01-27 16:17:40 +01:00
|
|
|
static enum ast_redir_type redir_tok_to_ast_type(enum token_type tok_type)
|
|
|
|
|
{
|
2026-01-27 19:56:33 +01:00
|
|
|
switch (tok_type)
|
2026-01-27 16:17:40 +01:00
|
|
|
{
|
2026-01-27 19:56:33 +01:00
|
|
|
case TOKEN_REDIR_LEFT:
|
|
|
|
|
return AST_REDIR_TYPE_LESS;
|
|
|
|
|
case TOKEN_REDIR_RIGHT:
|
|
|
|
|
return AST_REDIR_TYPE_GREAT;
|
2026-01-28 11:34:29 +01:00
|
|
|
case TOKEN_REDIR_DOUBLE_RIGHT:
|
|
|
|
|
return AST_REDIR_TYPE_DGREAT;
|
|
|
|
|
case TOKEN_REDIR_LEFT_RIGHT:
|
|
|
|
|
return AST_REDIR_TYPE_LESSGREAT;
|
|
|
|
|
case TOKEN_REDIR_LEFT_AMP:
|
|
|
|
|
return AST_REDIR_TYPE_LESSAND;
|
|
|
|
|
case TOKEN_REDIR_RIGHT_AMP:
|
|
|
|
|
return AST_REDIR_TYPE_GREATAND;
|
|
|
|
|
case TOKEN_REDIR_RIGHT_PIPE:
|
|
|
|
|
return AST_REDIR_TYPE_CLOBBER;
|
2026-01-27 19:56:33 +01:00
|
|
|
default:
|
|
|
|
|
return AST_REDIR_TYPE_NULL;
|
2026-01-27 16:17:40 +01:00
|
|
|
}
|
|
|
|
|
}
|
2026-01-27 16:05:11 +01:00
|
|
|
|
2026-01-24 15:34:10 +01:00
|
|
|
struct ast *parse_redirection(struct lexer_context *ctx)
|
|
|
|
|
{
|
|
|
|
|
struct token *token = PEEK_TOKEN();
|
2026-01-24 16:13:16 +01:00
|
|
|
int io_number = -1;
|
|
|
|
|
if (token->type == TOKEN_IONUMBER)
|
|
|
|
|
{
|
|
|
|
|
io_number = atoi(token->data);
|
2026-01-24 15:34:10 +01:00
|
|
|
POP_TOKEN();
|
|
|
|
|
token = PEEK_TOKEN();
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-27 16:05:11 +01:00
|
|
|
if (!is_token_redir(token))
|
2026-01-24 15:34:10 +01:00
|
|
|
{
|
2026-01-27 16:05:11 +01:00
|
|
|
perror("Syntax error: expected a redirection token but got something "
|
2026-01-27 19:56:33 +01:00
|
|
|
"else");
|
2026-01-24 15:34:10 +01:00
|
|
|
return NULL;
|
|
|
|
|
}
|
2026-01-27 16:17:40 +01:00
|
|
|
|
|
|
|
|
enum ast_redir_type redir_type = redir_tok_to_ast_type(token->type);
|
2026-01-24 16:13:16 +01:00
|
|
|
POP_TOKEN();
|
|
|
|
|
|
|
|
|
|
token = PEEK_TOKEN();
|
|
|
|
|
if (token->type != TOKEN_WORD)
|
|
|
|
|
{
|
2026-01-27 16:05:11 +01:00
|
|
|
perror("Syntax error: expected a word after redirection");
|
2026-01-24 16:13:16 +01:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
char *target = strdup(token->data);
|
|
|
|
|
POP_TOKEN();
|
|
|
|
|
|
2026-01-27 19:56:33 +01:00
|
|
|
return ast_create_redir(target, io_number, redir_type);
|
2026-01-24 15:34:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ast *parse_prefix(struct lexer_context *ctx)
|
|
|
|
|
{
|
2026-01-29 11:18:56 +01:00
|
|
|
struct token *token = TOKEN_PEEK();
|
|
|
|
|
if (token->type == TOKEN_ASSIGNMENT_WORD)
|
|
|
|
|
{
|
|
|
|
|
token = TOKEN_POP();
|
|
|
|
|
return ast_create_assignment_word(token->data);
|
|
|
|
|
}
|
|
|
|
|
else if (is_first(*token, RULE_REDIRECTION))
|
|
|
|
|
return parse_redirection(ctx);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
perror("Syntax error: expected a prefix (redirection or assignment)");
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2026-01-24 15:34:10 +01:00
|
|
|
}
|