Tatopenn's picture
Upload 20 files
4eff328 verified
# Copyright (c) 2026 Salvatore Pennacchio <jtatopenn@libero.it>
# Distributed under the Business Source License 1.1 (BSL 1.1)
# See LICENSE.md in the project root for full license terms.
import re
from typing import List, Dict, Optional, Tuple
from dataclasses import dataclass, field
import numpy as np
# ─────────────────────────────────────────────────────────────────────────────
# Data model
# ─────────────────────────────────────────────────────────────────────────────
@dataclass
class QASMCircuit:
"""
Parsed representation of an OpenQASM 2.0 / 3.0 circuit.
Attributes
----------
n_qubits : total qubit count declared in qreg / qubit statements
n_cbits : total classical bit count declared in creg / bit statements
ops : list of gate dicts β€” each dict has keys:
'type' : 'gate'
'name' : lowercase gate name (aliases resolved)
'qubits' : list[int] β€” absolute qubit indices
'params' : list[float] β€” evaluated rotation angles
"""
n_qubits: int = 0
n_cbits: int = 0
ops: List[Dict] = field(default_factory=list)
def to_tuples(self) -> List[Tuple]:
"""
Convert ops to the tuple format expected by DenseSVSimulator.run_circuit:
(name, qubit0[, qubit1, ...][, param0, ...])
BUG FIX (original): the original returned
(name,) + tuple(qubits) + tuple(params)
which placed params *after* qubits, but run_circuit expects
params interleaved or trailing depending on gate type.
For the standard (name, qubit, param) convention used throughout
the simulator, this ordering is correct β€” preserved here but
documented explicitly so callers know what to expect.
"""
out = []
for op in self.ops:
row = (op['name'],) + tuple(op['qubits']) + tuple(op['params'])
out.append(row)
return out
# ─────────────────────────────────────────────────────────────────────────────
# Parser
# ─────────────────────────────────────────────────────────────────────────────
class QASMParser:
"""
Robust OpenQASM 2.0 / 3.0 parser.
Supported features
------------------
- qreg / creg (QASM 2.0)
- qubit / bit (QASM 3.0)
- Parametric gates: rx, ry, rz, p, u1, u2, u3, cp, crz, ...
- Compound parameter expressions: pi/2, sqrt(2), cos(0.3), ...
- Block comments /* ... */ and line comments // ...
- Gate aliases: cu1→cp, u1→p, toffoli→ccx, cnot→cx, ...
- Range syntax q[0:3] expanded to individual qubits
- Bare register name (no index) resolved to qubit 0 of that register
- Silent fallback (0.0) for unparseable parameter expressions
"""
# ── compiled regexes ────────────────────────────────────────────
_RE_BLOCK_CMT = re.compile(r'/\*.*?\*/', re.DOTALL)
_RE_LINE_CMT = re.compile(r'//[^\n]*')
_RE_INDEX = re.compile(r'\[(\d+)\]')
_RE_RANGE = re.compile(r'^([a-zA-Z_]\w*)\[(\d+):(\d+)\]$') # q[0:3]
_RE_QREG2 = re.compile(r'^qreg\s+([a-zA-Z_]\w*)\s*\[(\d+)\]')
_RE_CREG2 = re.compile(r'^creg\s+([a-zA-Z_]\w*)\s*\[(\d+)\]')
_RE_QREG3 = re.compile(r'^qubit(?:\s*\[(\d+)\])?\s+([a-zA-Z_]\w*)')
_RE_CREG3 = re.compile(r'^bit(?:\s*\[(\d+)\])?\s+([a-zA-Z_]\w*)')
_RE_GATE_HEAD = re.compile(r'^([a-zA-Z_]\w*)(?:\((.*)\))?$')
# ── gate name aliases ────────────────────────────────────────────
_ALIAS: Dict[str, str] = {
'cu1': 'cp',
'u1': 'p',
'toffoli': 'ccx',
'fredkin': 'cswap',
'cnot': 'cx',
'not': 'x',
'id': 'i',
'cx': 'cx', # explicit identity mappings for safety
'cz': 'cz',
'ccx': 'ccx',
}
# ── statements to skip entirely ──────────────────────────────────
# BUG FIX (original): 'gate ' had a trailing space making it miss
# 'gate foo(...)' where the token is 'gate' followed by space.
# Using startswith on lowercased tokens is correct but the original
# also skipped 'def ' and 'for ' which are QASM 3.0 keywords β€”
# kept here for forward compatibility.
_SKIP = frozenset((
'openqasm', 'include', 'barrier', 'measure',
'reset', 'gate', 'def', 'if', 'for', 'while',
))
# ── safe math environment for eval() ────────────────────────────
_MATH_ENV: Dict = {
'__builtins__': {},
'pi': np.pi,
'tau': 2.0 * np.pi,
'euler': np.e,
'np': np,
'sin': np.sin, 'cos': np.cos, 'tan': np.tan,
'sqrt': np.sqrt, 'exp': np.exp, 'log': np.log,
'asin': np.arcsin,'acos': np.arccos,'atan': np.arctan,
'arcsin': np.arcsin,'arccos': np.arccos,'arctan': np.arctan,
'abs': abs, 'round': round,
}
# ────────────────────────────────────────────────────────────────
# Public interface
# ────────────────────────────────────────────────────────────────
def parse(self, qasm_str: str) -> QASMCircuit:
"""
Parse an OpenQASM 2.0 or 3.0 string into a QASMCircuit.
BUG FIX 1 (original): the original joined all lines with a single
space then split on ';'. Multi-line gate definitions (gate foo ...)
were not stripped before joining, causing 'gate foo ...' to appear
as a runnable instruction. Fixed by stripping comments *before*
joining and by using the frozenset _SKIP check on the first token.
BUG FIX 2 (original): bare register names (e.g. 'h q' instead of
'h q[0]') were silently dropped if the register had more than one
qubit, because qubit_map only stored 'name[0]' β†’ 0 for size-1
registers. Fixed: bare names always map to qubit 0 of that register
regardless of register size.
BUG FIX 3 (original): range syntax q[0:3] was never handled β€”
such tokens fell through to the digit-extraction fallback which
returned only the last digit. Fixed in _resolve_qubits.
"""
qubit_map: Dict[str, int] = {}
cbit_map: Dict[str, int] = {}
n_qubits = 0
n_cbits = 0
ops: List[Dict] = []
# ── strip comments ───────────────────────────────────────────
cleaned = self._RE_BLOCK_CMT.sub(' ', qasm_str)
cleaned = self._RE_LINE_CMT.sub(' ', cleaned)
# ── split into statements ─────────────────────────────────────
statements = [s.strip() for s in cleaned.split(';') if s.strip()]
for instr in statements:
# collapse internal whitespace runs to a single space
instr = re.sub(r'\s+', ' ', instr).strip()
if not instr:
continue
# first token (before any space or '(') for keyword detection
first_token = re.split(r'[\s(]', instr)[0].lower()
if first_token in self._SKIP:
continue
# ── qreg (QASM 2.0) ─────────────────────────────────────
m = self._RE_QREG2.match(instr)
if m:
reg_name, sz = m.group(1), int(m.group(2))
for i in range(sz):
qubit_map[f'{reg_name}[{i}]'] = n_qubits + i
qubit_map[reg_name] = n_qubits # bare name β†’ first qubit
n_qubits += sz
continue
# ── creg (QASM 2.0) ─────────────────────────────────────
m = self._RE_CREG2.match(instr)
if m:
reg_name, sz = m.group(1), int(m.group(2))
for i in range(sz):
cbit_map[f'{reg_name}[{i}]'] = n_cbits + i
cbit_map[reg_name] = n_cbits
n_cbits += sz
continue
# ── qubit (QASM 3.0) ─────────────────────────────────────
m = self._RE_QREG3.match(instr)
if m:
sz_s, reg_name = m.group(1), m.group(2)
sz = int(sz_s) if sz_s else 1
for i in range(sz):
qubit_map[f'{reg_name}[{i}]'] = n_qubits + i
qubit_map[reg_name] = n_qubits
n_qubits += sz
continue
# ── bit (QASM 3.0) ───────────────────────────────────────
m = self._RE_CREG3.match(instr)
if m:
sz_s, reg_name = m.group(1), m.group(2)
sz = int(sz_s) if sz_s else 1
for i in range(sz):
cbit_map[f'{reg_name}[{i}]'] = n_cbits + i
cbit_map[reg_name] = n_cbits
n_cbits += sz
continue
# ── gate application ─────────────────────────────────────
op = self._parse_gate(instr, qubit_map)
if op is not None:
ops.append(op)
# update n_qubits from seen qubit indices
# (handles circuits without explicit qreg declarations)
if op['qubits']:
n_qubits = max(n_qubits, max(op['qubits']) + 1)
return QASMCircuit(n_qubits, n_cbits, ops)
def validate(self, circ: QASMCircuit) -> Tuple[bool, str]:
"""Light structural validation β€” does not verify gate semantics."""
if circ.n_qubits <= 0:
return False, 'n_qubits must be > 0.'
if not circ.ops:
return False, 'No gate operations found in circuit.'
# check for out-of-range qubit references
for i, op in enumerate(circ.ops):
for q in op.get('qubits', []):
if not (0 <= q < circ.n_qubits):
return False, (
f"Gate '{op['name']}' at op[{i}] references "
f"qubit {q} but n_qubits={circ.n_qubits}.")
return True, 'OK'
# ────────────────────────────────────────────────────────────────
# Private helpers
# ────────────────────────────────────────────────────────────────
def _parse_gate(self,
instr: str,
qubit_map: Dict[str, int]) -> Optional[Dict]:
"""
Parse a single gate instruction into an op dict.
BUG FIX 4 (original): the original code had two independent
code paths for extracting param_str β€” one using _RE_GATE_HEAD
and one rescanning for '(' β€” that could disagree, leaving
param_str as the group(2) of an earlier (shorter) match while
paren_start/paren_end referred to a different range. Unified
into a single pass that:
1. finds the parameter parentheses (balanced),
2. extracts everything before '(' as the gate name,
3. extracts everything after the closing ')' as the qubit list.
BUG FIX 5 (original): split_at was found by scanning for the
first space at depth==0 *in the whole instruction*, so for
rx(pi/2) q[0]
split_at was -1 (no space outside parens in 'rx(pi/2)') and
rest was '' β€” dropping the qubit entirely. Fixed by splitting
on the space after the closing ')'.
"""
instr = instr.strip()
# ── locate parameter block '(...)' ───────────────────────────
paren_open = instr.find('(')
paren_close = -1
param_str = ''
if paren_open != -1:
depth = 0
for idx in range(paren_open, len(instr)):
if instr[idx] == '(':
depth += 1
elif instr[idx] == ')':
depth -= 1
if depth == 0:
paren_close = idx
break
if paren_close == -1:
# Unbalanced parentheses β€” skip this instruction
return None
param_str = instr[paren_open + 1 : paren_close].strip()
# gate_head = everything before '(', qubit_part = everything after ')'
gate_head = instr[:paren_open].strip()
qubit_part = instr[paren_close + 1:].strip()
else:
# No parameters: split on first whitespace
parts = instr.split(None, 1)
gate_head = parts[0]
qubit_part = parts[1] if len(parts) > 1 else ''
gate_name_raw = gate_head.strip().lower()
if not gate_name_raw:
return None
gate_name = self._ALIAS.get(gate_name_raw, gate_name_raw)
# ── parse parameters ─────────────────────────────────────────
params: List[float] = []
if param_str:
for tok in self._split_params(param_str):
tok = tok.strip()
if not tok:
continue
params.append(self._eval_param(tok))
# ── resolve qubits ───────────────────────────────────────────
qubits = self._resolve_qubits(
qubit_part.replace(' ', ''), qubit_map)
if not qubits:
return None
return {
'type': 'gate',
'name': gate_name,
'qubits': qubits,
'params': params,
}
def _eval_param(self, tok: str) -> float:
"""
Evaluate a parameter token to float.
Handles: numeric literals, pi, pi/2, sqrt(2), cos(0.3), etc.
Returns 0.0 on any evaluation error (silent fallback).
"""
try:
return float(eval(tok, self._MATH_ENV)) # noqa: S307
except Exception:
return 0.0
@staticmethod
def _split_params(s: str) -> List[str]:
"""
Split a comma-separated parameter string respecting nested
parentheses. e.g. 'pi/2, atan(1,0)' β†’ ['pi/2', 'atan(1,0)']
"""
tokens: List[str] = []
cur: List[str] = []
depth = 0
for ch in s:
if ch == '(':
depth += 1
cur.append(ch)
elif ch == ')':
depth -= 1
cur.append(ch)
elif ch == ',' and depth == 0:
tokens.append(''.join(cur).strip())
cur = []
else:
cur.append(ch)
if cur:
tokens.append(''.join(cur).strip())
return [t for t in tokens if t]
def _resolve_qubits(self,
s: str,
qmap: Dict[str, int]) -> List[int]:
"""
Resolve a comma-separated qubit argument string to absolute indices.
Handles
-------
- Indexed: q[0], q[1]
- Bare: q β†’ qmap['q'] (first qubit of that register)
- Range: q[0:3] β†’ [qmap['q[0]'], qmap['q[1]'], qmap['q[2]']]
BUG FIX 6 (original): range syntax q[0:3] was not handled and
fell through to the digit-extraction fallback, returning only
the last number found (e.g., 3 instead of [0,1,2]).
BUG FIX 7 (original): the fallback `digits = re.findall(r'\d+', tok)`
was used as a last resort β€” this could silently map unknown tokens
to arbitrary integers. Now the fallback is gated on the absence of
any letter character to avoid mapping named registers that are simply
not yet in qmap to wrong indices.
"""
out: List[int] = []
for tok in s.split(','):
tok = tok.strip()
if not tok:
continue
# ── range syntax: q[start:end] ───────────────────────────
m = self._RE_RANGE.match(tok)
if m:
base = m.group(1)
start = int(m.group(2))
end = int(m.group(3)) # exclusive upper bound
for i in range(start, end):
key = f'{base}[{i}]'
if key in qmap:
out.append(qmap[key])
continue
# ── direct map lookup ─────────────────────────────────────
if tok in qmap:
out.append(qmap[tok])
continue
# ── indexed: base[n] ─────────────────────────────────────
bracket = self._RE_INDEX.search(tok)
if bracket:
base = tok[:tok.index('[')]
key = f'{base}[{bracket.group(1)}]'
if key in qmap:
out.append(qmap[key])
continue
# index not in map β€” try numeric fallback
out.append(int(bracket.group(1)))
continue
# ── bare name not in map: try stripping to digits ─────────
# Only do this when the token contains no letters (pure numeric)
# to avoid misidentifying unknown register names.
digits = re.findall(r'\d+', tok)
if digits and not re.search(r'[a-zA-Z_]', tok):
out.append(int(digits[-1]))
return out