absyntax_utils/search_expression_type.cc
author Manuele Conti <conti.ma@alice.it>
Mon, 30 Jul 2012 22:59:00 +0200
changeset 622 eaa49b276e17
parent 350 2c3c4dc34979
permissions -rwxr-xr-x
Add check bison version in configure.ac file.
To build correctly matiec we need bison greater or equals than 2.4 version.
Now the "configure" script is able to check if system has correctly requirements.
/*
 *  matiec - a compiler for the programming languages defined in IEC 61131-3
 *
 *  Copyright (C) 2003-2011  Mario de Sousa (msousa@fe.up.pt)
 *  Copyright (C) 2007-2011  Laurent Bessard and Edouard Tisserant
 *
 *  This program 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 of the License, or
 *  (at your option) any later version.
 *
 *  This program 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 this program.  If not, see <http://www.gnu.org/licenses/>.
 *
 *
 * This code is made available on the understanding that it will not be
 * used in safety-critical situations without a full and competent review.
 */

/*
 * An IEC 61131-3 compiler.
 *
 * Based on the
 * FINAL DRAFT - IEC 61131-3, 2nd Ed. (2001-12-10)
 *
 */


/* Determine the data type of an ST expression.
 * A reference to the relevant type definition is returned.
 *
 * For example:
 *       2 + 3       -> returns reference to a int_type_name_c object.
 *       22.2 - 5    -> returns reference to a real_type_name_c object.
 *       etc...
 */



#include "absyntax_utils.hh"
#include <typeinfo>

search_expression_type_c::search_expression_type_c(symbol_c *search_scope) {
  search_varfb_instance_type = new search_varfb_instance_type_c(search_scope);
}

search_expression_type_c::~search_expression_type_c(void) {
  delete search_varfb_instance_type;
}

/* A helper function... */
bool search_expression_type_c::is_bool_type(symbol_c *type_symbol) {
  bool_type_name_c tt;
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(safebool_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(bool_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(boolean_true_c))       {return true;}
  if (typeid(*type_symbol) == typeid(boolean_false_c))      {return true;}
  return false;
}

/* A helper function... */
bool search_expression_type_c::is_time_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(time_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(date_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(tod_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(dt_type_name_c))       {return true;}
  if (typeid(*type_symbol) == typeid(safetime_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(safedate_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(safetod_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safedt_type_name_c))   {return true;}
  return false;
}

/* A helper function... */
bool search_expression_type_c::is_string_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(string_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(wstring_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(safestring_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safewstring_type_name_c)) {return true;}
  return false;
}

/* A helper function... */
bool search_expression_type_c::is_literal_integer_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(integer_c))        {return true;}
  if (typeid(*type_symbol) == typeid(neg_integer_c))    {return true;}
  if (typeid(*type_symbol) == typeid(binary_integer_c)) {return true;}
  if (typeid(*type_symbol) == typeid(octal_integer_c))  {return true;}
  if (typeid(*type_symbol) == typeid(hex_integer_c))    {return true;}
  return false;
}

/* A helper function... */
bool search_expression_type_c::is_integer_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(sint_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(int_type_name_c))       {return true;}
  if (typeid(*type_symbol) == typeid(dint_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(lint_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(usint_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(uint_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(udint_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(ulint_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(safesint_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safeint_type_name_c))   {return true;}
  if (typeid(*type_symbol) == typeid(safedint_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safelint_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safeusint_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(safeuint_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safeudint_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(safeulint_type_name_c)) {return true;}
  return is_literal_integer_type(type_symbol);
}

/* A helper function... */
bool search_expression_type_c::is_literal_real_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(real_c))     {return true;}
  if (typeid(*type_symbol) == typeid(neg_real_c)) {return true;}
  return false;
}

/* A helper function... */
bool search_expression_type_c::is_real_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(real_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(lreal_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(safereal_type_name_c))  {return true;} 
  if (typeid(*type_symbol) == typeid(safelreal_type_name_c)) {return true;}  
  return is_literal_real_type(type_symbol);
}

bool search_expression_type_c::is_num_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  return is_real_type(type_symbol) || is_integer_type(type_symbol);
}

bool search_expression_type_c::is_nbinary_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
  if (typeid(*type_symbol) == typeid(byte_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(word_type_name_c))      {return true;}
  if (typeid(*type_symbol) == typeid(dword_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(lword_type_name_c))     {return true;}
  if (typeid(*type_symbol) == typeid(safebyte_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safeword_type_name_c))  {return true;}
  if (typeid(*type_symbol) == typeid(safedword_type_name_c)) {return true;}
  if (typeid(*type_symbol) == typeid(safelword_type_name_c)) {return true;}
  return is_literal_integer_type(type_symbol);
}

bool search_expression_type_c::is_binary_type(symbol_c *type_symbol) {
  if (type_symbol == NULL) {return true;}
//   if (typeid(*type_symbol) == typeid(bool_type_name_c))     {return true;}
//   if (typeid(*type_symbol) == typeid(safebool_type_name_c)) {return true;}
  return (is_nbinary_type(type_symbol) || is_bool_type(type_symbol));
}

bool search_expression_type_c::is_same_type(symbol_c *first_type, symbol_c *second_type) {
  if (first_type == NULL || second_type == NULL) {return true;}
  if (typeid(*first_type) == typeid(*second_type)) {return true;}
  if (is_bool_type(first_type)            && is_bool_type(second_type))            {return true;}
  if (is_integer_type(first_type)         && is_literal_integer_type(second_type)) {return true;}
  if (is_literal_integer_type(first_type) && is_integer_type(second_type))         {return true;}
  if (is_binary_type(first_type)          && is_literal_integer_type(second_type)) {return true;}
  if (is_literal_integer_type(first_type) && is_binary_type(second_type))          {return true;}
  if (is_real_type(first_type)            && is_literal_real_type(second_type))    {return true;}
  if (is_literal_real_type(first_type)    && is_real_type(second_type))            {return true;}
  return false;
}

symbol_c* search_expression_type_c::common_type(symbol_c *first_type, symbol_c *second_type) {
  if (first_type == NULL && second_type == NULL) {return NULL;}
  if (first_type == NULL) {return second_type;}
  if (second_type == NULL) {return first_type;}
  if (typeid(*first_type) == typeid(*second_type)) {return first_type;}
  if (is_integer_type(first_type)         && is_literal_integer_type(second_type)) {return first_type;}
  if (is_literal_integer_type(first_type) && is_integer_type(second_type))         {return second_type;}
  if (is_binary_type(first_type)          && is_literal_integer_type(second_type)) {return first_type;}
  if (is_literal_integer_type(first_type) && is_binary_type(second_type))          {return second_type;}
  if (is_real_type(first_type)            && is_literal_real_type(second_type))    {return first_type;}
  if (is_literal_real_type(first_type)    && is_real_type(second_type))            {return second_type;}
  return NULL;
}

symbol_c *search_expression_type_c::default_literal_type(symbol_c* symbol) {
  if (is_literal_integer_type(symbol)) {
    return (symbol_c *)&search_constant_type_c::lint_type_name;
  }
  else if (is_literal_real_type(symbol)) {
    return (symbol_c *)&search_constant_type_c::lreal_type_name;
  }
  return symbol;
}


/* A helper function... */
void *search_expression_type_c::compute_boolean_expression(symbol_c *left_type, symbol_c *right_type) {
  if (!is_same_type(left_type, right_type))
    ERROR;
  if (!is_bool_type(left_type) && !is_binary_type(left_type))
    ERROR;
  if (is_literal_integer_type(left_type)) {return (void *)right_type;}
  else {return (void *)left_type;}
}

/* A helper function... */
void *search_expression_type_c::compute_numeric_expression(symbol_c *left_type, symbol_c *right_type) {
  if (!is_same_type(left_type, right_type))
    ERROR;
  if (!is_integer_type(left_type) && !is_real_type(left_type))
    ERROR;
  if (is_literal_integer_type(left_type) || is_literal_real_type(left_type)) {return (void *)right_type;}
  else {return (void *)left_type;}
  return NULL;
}

/* a helper function... */
symbol_c *search_expression_type_c::base_type(symbol_c *symbol) {
  return (symbol_c *)symbol->accept(search_base_type);
}

/*********************/
/* B 1.4 - Variables */
/*********************/

void *search_expression_type_c::visit(symbolic_variable_c *symbol) {
  symbol_c *res;
  
  /* Nope, now we assume it is a variable, and determine its type... */
  res = search_varfb_instance_type->get_basetype_decl(symbol);
  if (NULL != res) return res;
  
  return NULL;
}

/********************************************/
/* B 1.4.1 - Directly Represented Variables */
/********************************************/
void *search_expression_type_c::visit(direct_variable_c *symbol) {
  symbol_c *res;
  
  /* Nope, now we assume it is a variable, and determine its type... */
  res = search_varfb_instance_type->get_basetype_decl(symbol);
  if (NULL != res) return res;
  
  return NULL;
}

/*************************************/
/* B 1.4.2 - Multi-element variables */
/*************************************/

void *search_expression_type_c::visit(array_variable_c *symbol) {
  symbol_c *res;
  
  /* Nope, now we assume it is a variable, and determine its type... */
  res = search_varfb_instance_type->get_basetype_decl(symbol);
  if (NULL != res) return res;
  
  return NULL;
}

void *search_expression_type_c::visit(structured_variable_c *symbol) {
  symbol_c *res;
  
  /* Nope, now we assume it is a variable, and determine its type... */
  res = search_varfb_instance_type->get_basetype_decl(symbol);
  if (NULL != res) return res;
  
  return NULL;
}

/***************************************/
/* B.3 - Language ST (Structured Text) */
/***************************************/
/***********************/
/* B 3.1 - Expressions */
/***********************/
void *search_expression_type_c::visit(or_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  return compute_boolean_expression(left_type, right_type);
}

void *search_expression_type_c::visit(xor_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  return compute_boolean_expression(left_type, right_type);
}

void *search_expression_type_c::visit(and_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  return compute_boolean_expression(left_type, right_type); 
}

void *search_expression_type_c::visit(equ_expression_c *symbol) {return (void *)&bool_type_name;}
void *search_expression_type_c::visit(notequ_expression_c *symbol) {return (void *)&bool_type_name;}
void *search_expression_type_c::visit(lt_expression_c *symbol) {return (void *)&bool_type_name;}
void *search_expression_type_c::visit(gt_expression_c *symbol) {return (void *)&bool_type_name;}
void *search_expression_type_c::visit(le_expression_c *symbol) {return (void *)&bool_type_name;}
void *search_expression_type_c::visit(ge_expression_c *symbol) {return (void *)&bool_type_name;}

void *search_expression_type_c::visit(add_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  if (typeid(*left_type) == typeid(time_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&time_type_name;}
  if (typeid(*left_type) == typeid(tod_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&tod_type_name;}
  if (typeid(*left_type) == typeid(dt_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&dt_type_name;}
  return compute_numeric_expression(left_type, right_type);
}

void *search_expression_type_c::visit(sub_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  if (typeid(*left_type) == typeid(time_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&time_type_name;}
  if (typeid(*left_type) == typeid(date_type_name_c) && typeid(*right_type) == typeid(date_type_name_c)) {return (void *)&time_type_name;}
  if (typeid(*left_type) == typeid(tod_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&tod_type_name;}
  if (typeid(*left_type) == typeid(tod_type_name_c) && typeid(*right_type) == typeid(tod_type_name_c)) {return (void *)&time_type_name;}
  if (typeid(*left_type) == typeid(dt_type_name_c) && typeid(*right_type) == typeid(time_type_name_c)) {return (void *)&dt_type_name;}
  if (typeid(*left_type) == typeid(dt_type_name_c) && typeid(*right_type) == typeid(dt_type_name_c)) {return (void *)&time_type_name;}
  return compute_numeric_expression(left_type, right_type);
}

void *search_expression_type_c::visit(mul_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  if (typeid(*left_type) == typeid(time_type_name_c) && is_num_type(right_type)) {
      return (void *)&time_type_name;
  }
  return compute_numeric_expression(left_type, right_type);
}

void *search_expression_type_c::visit(div_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  if (typeid(*left_type) == typeid(time_type_name_c) && is_num_type(right_type)){
      return (void *)&time_type_name;
  }
  return compute_numeric_expression(left_type, right_type);
}

void *search_expression_type_c::visit(mod_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  return compute_numeric_expression(left_type, right_type);
}

void *search_expression_type_c::visit(power_expression_c *symbol) {
  symbol_c *left_type = base_type((symbol_c *)symbol->l_exp->accept(*this));
  symbol_c *right_type = base_type((symbol_c *)symbol->r_exp->accept(*this));
  if (is_real_type(left_type) && is_num_type(right_type)) {
      return (void *)left_type;
  }
  ERROR;
  return NULL;
}

void *search_expression_type_c::visit(neg_expression_c *symbol) {
  symbol_c *exp_type = base_type((symbol_c *)symbol->exp->accept(*this));
  if (is_num_type(exp_type) || typeid(*exp_type) == typeid(time_type_name_c)){
      return (void *)exp_type;
  }
  ERROR;
  return NULL;
}

void *search_expression_type_c::visit(not_expression_c *symbol) {
  symbol_c *exp_type = base_type((symbol_c *)symbol->exp->accept(*this));
  return compute_boolean_expression(exp_type, exp_type);
}

void *search_expression_type_c::visit(function_invocation_c *symbol) {
  function_declaration_c *f_decl = (function_declaration_c *)symbol->called_function_declaration;
  if (f_decl == NULL) ERROR;
  return base_type(f_decl->type_name);
}

/*bool_type_name_c     search_expression_type_c::bool_type_name;*/