|
| 1 | +"""Host-side software for ILA""" |
| 2 | +from argparse import ArgumentParser |
| 3 | +from serial import Serial |
| 4 | +from vcd import VCDWriter |
| 5 | + |
| 6 | + |
| 7 | +def parse_layout(layout_str): |
| 8 | + """Parse signal layout string into list of (name, width) tuples. |
| 9 | + |
| 10 | + Args: |
| 11 | + layout_str: String like 'data_in:10,trigger:1,address:8' |
| 12 | + |
| 13 | + Returns: |
| 14 | + List of (signal_name, width) tuples |
| 15 | + |
| 16 | + Raises: |
| 17 | + ValueError: If layout string is malformed |
| 18 | + """ |
| 19 | + signals = [] |
| 20 | + if not layout_str.strip(): |
| 21 | + raise ValueError("Layout string cannot be empty") |
| 22 | + |
| 23 | + for signal_def in layout_str.split(','): |
| 24 | + signal_def = signal_def.strip() |
| 25 | + if ':' not in signal_def: |
| 26 | + raise ValueError(f"Invalid signal definition '{signal_def}'. Expected format 'name:width'") |
| 27 | + |
| 28 | + name, width_str = signal_def.split(':', 1) |
| 29 | + name = name.strip() |
| 30 | + width_str = width_str.strip() |
| 31 | + |
| 32 | + if not name: |
| 33 | + raise ValueError(f"Signal name cannot be empty in '{signal_def}'") |
| 34 | + |
| 35 | + try: |
| 36 | + width = int(width_str) |
| 37 | + if width <= 0: |
| 38 | + raise ValueError(f"Signal width must be positive, got {width} for '{name}'") |
| 39 | + except ValueError as e: |
| 40 | + if "invalid literal" in str(e): |
| 41 | + raise ValueError(f"Invalid width '{width_str}' for signal '{name}'. Width must be a positive integer") |
| 42 | + raise |
| 43 | + |
| 44 | + signals.append((name, width)) |
| 45 | + |
| 46 | + return signals |
| 47 | + |
| 48 | + |
| 49 | +def extract_signal_value(data_value, bit_offset, width): |
| 50 | + """Extract a signal value from the packed data. |
| 51 | + |
| 52 | + Args: |
| 53 | + data_value: Full packed data value |
| 54 | + bit_offset: Starting bit position (LSB = 0) |
| 55 | + width: Number of bits to extract |
| 56 | + |
| 57 | + Returns: |
| 58 | + Extracted signal value |
| 59 | + """ |
| 60 | + mask = (1 << width) - 1 |
| 61 | + return (data_value >> bit_offset) & mask |
| 62 | + |
| 63 | + |
| 64 | +if __name__ == "__main__": |
| 65 | + parser = ArgumentParser(description="ILA Capture Tool") |
| 66 | + parser.add_argument("port", help="Serial port for ILA data") |
| 67 | + parser.add_argument("output", help="Output VCD file") |
| 68 | + parser.add_argument("--baudrate", type=int, default=115200, help="Baud rate for serial communication (default: 115200)") |
| 69 | + parser.add_argument("--depth", type=int, required=True, help="Number of samples captured by ILA") |
| 70 | + parser.add_argument("--layout", type=str, required=True, help="Signal layout description (e.g. 'data_in:10,trigger:1,address:8')") |
| 71 | + args = parser.parse_args() |
| 72 | + |
| 73 | + # Parse and validate the layout |
| 74 | + try: |
| 75 | + signals = parse_layout(args.layout) |
| 76 | + except ValueError as e: |
| 77 | + print(f"Error parsing layout: {e}") |
| 78 | + exit(1) |
| 79 | + |
| 80 | + # Calculate data width from layout |
| 81 | + data_width = sum(width for _, width in signals) |
| 82 | + |
| 83 | + print(f"Parsed layout: {signals}") |
| 84 | + print(f"Inferred data width: {data_width} bits") |
| 85 | + print(f"Waiting for {args.depth} samples of {data_width}-bit data from {args.port}") |
| 86 | + |
| 87 | + try: |
| 88 | + with Serial(args.port, args.baudrate, timeout=None) as ser, open(args.output, "w") as vcd_file: |
| 89 | + with VCDWriter(vcd_file, timescale="1 ns") as vcd: |
| 90 | + # Register clock signal |
| 91 | + clk_signal = vcd.register_var("ila", "clk", "wire", size=1) |
| 92 | + |
| 93 | + # Register all signals in the VCD |
| 94 | + vcd_signals = [] |
| 95 | + for name, width in signals: |
| 96 | + vcd_signal = vcd.register_var("ila", name, "wire", size=width) |
| 97 | + vcd_signals.append(vcd_signal) |
| 98 | + |
| 99 | + # Initialize clock signal to low |
| 100 | + vcd.change(clk_signal, 0, 0) |
| 101 | + |
| 102 | + # Capture and decode samples |
| 103 | + bytes_to_read = (data_width + 7) // 8 # Round up to nearest byte |
| 104 | + print(f"Waiting for data on {args.port}... (Press Ctrl+C to abort)") |
| 105 | + |
| 106 | + for sample_index in range(args.depth): |
| 107 | + # Read raw data from serial port - this will block until data is available |
| 108 | + raw_data = ser.read(bytes_to_read) |
| 109 | + |
| 110 | + if len(raw_data) < bytes_to_read: |
| 111 | + print(f"Incomplete data received at sample {sample_index}. Expected {bytes_to_read} bytes, got {len(raw_data)}. Exiting.") |
| 112 | + break |
| 113 | + |
| 114 | + # Convert bytes to integer (little-endian) |
| 115 | + data_value = int.from_bytes(raw_data, byteorder='little') |
| 116 | + |
| 117 | + # Extract and log individual signal values |
| 118 | + bit_offset = 0 |
| 119 | + timestamp = sample_index * 1000 # 1ns timestep, 1us per sample |
| 120 | + |
| 121 | + # Generate clock signal - rising edge at start of each sample |
| 122 | + vcd.change(clk_signal, timestamp, 1) |
| 123 | + |
| 124 | + for (name, width), vcd_signal in zip(signals, vcd_signals): |
| 125 | + signal_value = extract_signal_value(data_value, bit_offset, width) |
| 126 | + vcd.change(vcd_signal, timestamp, signal_value) |
| 127 | + bit_offset += width |
| 128 | + |
| 129 | + # Falling edge at middle of sample period |
| 130 | + vcd.change(clk_signal, timestamp + 500, 0) |
| 131 | + |
| 132 | + if sample_index % 100 == 0: # Progress indicator |
| 133 | + print(f"Processed {sample_index + 1}/{args.depth} samples") |
| 134 | + |
| 135 | + print(f"Data capture complete. Output written to {args.output}") |
| 136 | + |
| 137 | + except KeyboardInterrupt: |
| 138 | + print(f"\nCapture interrupted by user. Partial data written to {args.output}") |
| 139 | + except Exception as e: |
| 140 | + print(f"Error during capture: {e}") |
| 141 | + exit(1) |
0 commit comments