Skip to content

Commit

Permalink
Add support for ENV comments brackets
Browse files Browse the repository at this point in the history
Match all operations between "type": "ACTION_BEGIN" and "type": "ACTION_END" by name tag and create a new function block for them
See issue #25
  • Loading branch information
vmordan authored Jun 14, 2024
1 parent f9f3230 commit e760872
Show file tree
Hide file tree
Showing 2 changed files with 75 additions and 2 deletions.
2 changes: 1 addition & 1 deletion scripts/mea/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -154,7 +154,7 @@ def __convert_call_tree_filter(error_trace: dict, args: dict = None) -> list:
# TODO: check this in core (one node for call and return edges).
double_funcs = {}
for edge in error_trace['edges']:
if 'entry_point' in edge:
if 'entry_point' in edge or 'ignore MEA' in edge:
continue
if 'enter' in edge and 'return' in edge:
double_funcs[edge['enter']] = edge['return']
Expand Down
75 changes: 74 additions & 1 deletion scripts/mea/et/internal_witness.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@

WITNESS_TYPE_VIOLATION = "violation"
WITNESS_TYPE_CORRECTNESS = "correctness"
MAX_SEARCH_DISTANCE_FOR_EMG_COMMENT = 20


# Capitalize first letters of attribute names.
Expand Down Expand Up @@ -65,6 +66,7 @@ def __init__(self, logger):
self._spec_funcs = {}
self._env_models = {}
self._env_models_json = {}
self._env_funcs_openers = {}
self._notes = {}
self._asserts = {}
self._actions = []
Expand Down Expand Up @@ -251,6 +253,9 @@ def process_verifier_notes(self):
self.is_notes = True

warn_edges = []
last_opened_line = 0
counter = 0
new_edges = {}
for edge in self._edges:
if 'warn' in edge:
warn_edges.append(edge['warn'])
Expand All @@ -276,6 +281,44 @@ def process_verifier_notes(self):
f"{self._resolve_function(func_id)}'")
edge['env'] = self.process_comment(env_note)

if file_id in self._env_funcs_openers:
if last_opened_line > 0 and start_line - last_opened_line <= MAX_SEARCH_DISTANCE_FOR_EMG_COMMENT:
self._logger.debug(f"ENV comments brackets: search from {last_opened_line} to {start_line}")
for checked_line in range(last_opened_line, start_line):
if checked_line in self._env_funcs_openers[file_id]:
self._logger.debug(f"ENV comments brackets: found termination bracket on {checked_line} "
f"with last checked {last_opened_line}")
new_func_name = self._env_funcs_openers[file_id][checked_line]['name']
new_edges[counter] = {
'return': self.add_function(new_func_name),
'start line': checked_line,
'file': file_id,
'source': "",
'ignore MEA': True,
}
if 'thread' in edge:
new_edges[counter]['thread'] = edge['thread']
del self._env_funcs_openers[file_id][checked_line]
last_opened_line = 0
break
if start_line in self._env_funcs_openers[file_id]:
if self._env_funcs_openers[file_id][start_line]['type'] == "ACTION_BEGIN":
last_opened_line = start_line - 1
self._logger.debug(f"ENV comments brackets: found start bracket on {start_line}")
new_func_name = self._env_funcs_openers[file_id][start_line]['name']
specific_comment = self._env_funcs_openers[file_id][start_line]['comment']
new_edges[counter] = {
'enter': self.add_function(new_func_name),
'start line': start_line,
'file': file_id,
'env': specific_comment or new_func_name,
'source': f"{new_func_name}()",
'ignore MEA': True
}
if 'thread' in edge:
new_edges[counter]['thread'] = edge['thread']
del self._env_funcs_openers[file_id][start_line]

if file_id in self._notes and start_line in self._notes[file_id]:
note = self._notes[file_id][start_line]
self._logger.debug(f"Add note {note} for statement from '{file}:{start_line}'")
Expand All @@ -285,7 +328,7 @@ def process_verifier_notes(self):
comment = env["comment"]
relevant = env.get("relevant", False)

#TODO add remaining
# TODO add remaining
self._logger.debug(f"Add EMG comment '{comment}' for operation from '{file}:{start_line}'")
self._logger.debug(f"Comment argument: relevant={relevant}")
edge['env'] = self.process_comment(comment)
Expand All @@ -302,6 +345,27 @@ def process_verifier_notes(self):
if spec_func in edge['source']:
edge['note'] = self.process_comment(note)
break
counter += 1

stack = []
for index, edge in reversed(sorted(new_edges.items())):
if not stack and 'enter' in edge:
# Ignore enter on empty stack
continue
if 'return' in edge:
stack.append((edge['return'], index))
if 'enter' in edge:
last_elem_on_stack, last_index = stack.pop()
if not last_elem_on_stack == edge['enter']:
self._logger.warning(f"ENV comments brackets: found different element on env functions stack: "
f"{edge['enter']} instead of {last_elem_on_stack}")
else:
for cur_index in range(index, last_index):
if 'env' in self._edges[cur_index]:
del self._edges[cur_index]
self._logger.debug(
f"ENV comments brackets: remove ENV comment on the line {self._edges[cur_index]}")
self._edges.insert(index, edge)

if not warn_edges and self.witness_type == WITNESS_TYPE_VIOLATION:
if self._edges:
Expand Down Expand Up @@ -344,6 +408,15 @@ def _parse_model_comments(self):
if file_id not in self._env_models_json:
self._env_models_json[file_id] = {}
self._env_models_json[file_id][line + 1] = data
if 'name' in data:
func_data = {
'type': data['type'],
'name': data['name'],
'comment': data.get('comment', None)
}
if file_id not in self._env_funcs_openers:
self._env_funcs_openers[file_id] = {}
self._env_funcs_openers[file_id][line + 1] = func_data

# Match rest comments
match = re.search(
Expand Down

0 comments on commit e760872

Please sign in to comment.