{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "b816c094",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "b816c094",
    "outputId": "e1a4c487-6ece-4333-ed4b-197ffad71c87"
   },
   "outputs": [],
   "source": [
    "# !pip install z3-solver\n",
    "# !pip install spikingjelly"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5626942e",
   "metadata": {
    "id": "5626942e"
   },
   "outputs": [],
   "source": [
    "# ML + NN imports\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torchvision\n",
    "from torchvision import datasets, transforms\n",
    "from torchvision.utils import make_grid\n",
    "from torch.utils.data import DataLoader, Subset, RandomSampler\n",
    "from torch.cuda import amp\n",
    "from sklearn.model_selection import train_test_split"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "35c18adb",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch.nn.functional as F"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "703fde33",
   "metadata": {
    "id": "703fde33"
   },
   "outputs": [],
   "source": [
    "# Utils imports\n",
    "import os\n",
    "import time\n",
    "import sys\n",
    "import datetime\n",
    "import argparse\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.patches as patches\n",
    "import math"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "8aa790d1",
   "metadata": {
    "id": "8aa790d1"
   },
   "outputs": [],
   "source": [
    "# Spiking NN imports\n",
    "from spikingjelly.activation_based import neuron, functional, surrogate, layer, encoding\n",
    "from spikingjelly import visualizing"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "20f384a3",
   "metadata": {
    "id": "20f384a3"
   },
   "outputs": [],
   "source": [
    "# Python solver import\n",
    "from z3 import *\n",
    "from itertools import combinations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "2ac863e5",
   "metadata": {
    "id": "2ac863e5"
   },
   "outputs": [],
   "source": [
    "from tqdm import tqdm\n",
    "import yaml"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6792de24",
   "metadata": {
    "id": "6792de24"
   },
   "outputs": [],
   "source": [
    "# args={}\n",
    "# kwargs={}\n",
    "\n",
    "# args['device'] = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "# args['batch_size']=10\n",
    "# args['test_batch_size']=10\n",
    "# args['start_epoch'] = 0\n",
    "# args['epochs']=50\n",
    "# args['lr']=0.001\n",
    "\n",
    "# args['torch-seed'] = 99 #random seed\n",
    "# args['seed-sklearn'] = 54\n",
    "\n",
    "# args['log_interval'] = 10\n",
    "# args['cuda'] = True\n",
    "\n",
    "# args['T'] = 20 # Maximum time step for the firing of neurons\n",
    "# args['input-size'] = 28*28 # Set the input size\n",
    "# args['num_classes'] = 3  # Number of classes in MNIST\n",
    "# args['hidden_neurons'] = 2*4 # Number of neurons in the single hidden neuron\n",
    "# args['spiking_threshold'] = 1 # Spiking threshold for Integrate-Fire neurons\n",
    "\n",
    "# args['bin_thresh'] = 1.0 # Binarizing threshold for the weights\n",
    "\n",
    "# args['img_ind'] = 62 # Change image to compute explanation\n",
    "# args['time_slice'] = 0 # Change time slice to compute explanation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "b0c53153",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch._C.Generator at 0x11441c890>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Save to YAML file\n",
    "args_file_path = '3_digit_models/'\n",
    "# Save to YAML file\n",
    "yaml_file_name = 'final_exps/3-digit/args_8_[1, 5, 9]_(3.5348103046417236, 1.0024094581604004).yaml'\n",
    "with open(yaml_file_name, \"r\") as file:\n",
    "    args = yaml.safe_load(file)\n",
    "\n",
    "torch.manual_seed(args['torch-seed'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "24972415",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'T': 1,\n",
       " 'alpha_l2': 0.0001,\n",
       " 'batch_size': 32,\n",
       " 'epochs': 50,\n",
       " 'epsilon': 1e-05,\n",
       " 'hidden_neurons': 8,\n",
       " 'img_ind': 91,\n",
       " 'initial_thresh': 3.0,\n",
       " 'input-size': 784,\n",
       " 'log_interval': 10,\n",
       " 'lr': 0.1,\n",
       " 'lr_decay': 0.001,\n",
       " 'lr_th_decay': 1,\n",
       " 'lr_threshold': 0.01,\n",
       " 'momentum': 0.5,\n",
       " 'num_classes': 3,\n",
       " 'seed-sklearn': 54,\n",
       " 'start_epoch': 0,\n",
       " 'test_batch_size': 10,\n",
       " 'torch-seed': 99,\n",
       " 'usage_device': True,\n",
       " 'v_thresh_1': 2.0,\n",
       " 'v_thresh_2': 1.0,\n",
       " 'weight_decay': 0}"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Set device\n",
    "args['device'] =  torch.device('mps' if torch.backends.mps.is_available() else 'cpu')\n",
    "args"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bf1ef549",
   "metadata": {
    "id": "bf1ef549"
   },
   "source": [
    "# Define the Data Loader class for MNIST\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "bfaf7de2",
   "metadata": {
    "id": "bfaf7de2"
   },
   "outputs": [],
   "source": [
    "# Map digits {1, 5, 9} to the indexes {0, 1, 2}\n",
    "label_list = torch.tensor([1,5,9])\n",
    "label_map = {label_list[i].item() : i for i in range(len(label_list))}\n",
    "label_map_inverse = { i : label_list[i].item() for i in range(len(label_list))}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "11caa18f",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "11caa18f",
    "outputId": "6c1ea4a3-c795-4834-b590-f072b66a71ec"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of training sample frames 18112\n",
      "Number of validation sample frames 2125\n",
      "Number of test sample frames 911\n"
     ]
    }
   ],
   "source": [
    "# LOAD MNIST DATASET\n",
    "\n",
    "# download and transform train dataset\n",
    "train_dataset = datasets.MNIST('../mnist_data', download=True, train=True, transform=transforms.Compose([ transforms.ToTensor() ]))\n",
    "\n",
    "val_test_dataset = datasets.MNIST('../mnist_data', download=True, train=False, transform=transforms.Compose([ transforms.ToTensor() ]))\n",
    "\n",
    "\n",
    "# Convert dataset labels to a PyTorch tensor\n",
    "train_labels = train_dataset.targets.clone()\n",
    "val_test_labels = val_test_dataset.targets.clone()\n",
    "\n",
    "# Define your chosen digits\n",
    "\n",
    "\n",
    "# Create filter indices for training and testing labels\n",
    "filter_indices_train = (train_labels.unsqueeze(1) == label_list).any(dim=1)\n",
    "filter_indices_test = (val_test_labels.unsqueeze(1) == label_list).any(dim=1)\n",
    "\n",
    "# Apply filters to datasets\n",
    "train_dataset.data, train_dataset.targets = train_dataset.data[filter_indices_train], train_dataset.targets[filter_indices_train]\n",
    "val_test_dataset.data, val_test_dataset.targets = val_test_dataset.data[filter_indices_test], val_test_dataset.targets[filter_indices_test]\n",
    "\n",
    "\n",
    "# split the validation and test datasets\n",
    "val_idx, test_idx = train_test_split(list(range(len(val_test_dataset))), test_size=0.3, random_state = args['seed-sklearn'])\n",
    "\n",
    "val_dataset = Subset(val_test_dataset, val_idx)\n",
    "test_dataset = Subset(val_test_dataset, test_idx)\n",
    "\n",
    "# Define train, validation and test data loader\n",
    "train_loader = DataLoader(train_dataset, batch_size=args['batch_size'], shuffle=True)\n",
    "val_loader = DataLoader(val_dataset, batch_size=args['test_batch_size'], shuffle=False)\n",
    "test_loader = DataLoader(test_dataset, batch_size=args['test_batch_size'], shuffle=False)\n",
    "\n",
    "print(\"Number of training sample frames\", len(train_dataset))\n",
    "print(\"Number of validation sample frames\", len(val_dataset))\n",
    "print(\"Number of test sample frames\", len(test_dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b0b1de61",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 195
    },
    "id": "b0b1de61",
    "outputId": "cdb13e55-ce2d-4363-cdec-731253741203"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 800x800 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualise MNIST images\n",
    "def show_images(images, nmax=64):\n",
    "    fig, ax = plt.subplots(figsize=(8, 8))\n",
    "    ax.set_xticks([]); ax.set_yticks([])\n",
    "    ax.imshow(make_grid((images.detach()[:nmax]), nrow=8).permute(1, 2, 0))\n",
    "\n",
    "def show_batch(dl, nmax=64):\n",
    "    for images, _ in dl:\n",
    "        show_images(images, nmax)\n",
    "        break\n",
    "\n",
    "show_batch(test_loader)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f8dec116",
   "metadata": {
    "id": "f8dec116"
   },
   "source": [
    "### Encode MNIST images to spikes and visualise"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "933044f6",
   "metadata": {
    "id": "933044f6"
   },
   "outputs": [],
   "source": [
    "# Define Poisson encoder for visualisation\n",
    "encoder_sample = encoding.PoissonEncoder() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "d0b3c0d1",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 902
    },
    "id": "d0b3c0d1",
    "outputId": "4044fbbf-7a6b-46e3-b909-37b3930d5880"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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L9WpU+eCfnltezF/D1Xna5fkqP98Pf/8eVSqVMnRF8DlIxgBV7iWLVsqKipK7dq1U/369S+4lfyfDyjMj/r16ztuf2/Tpo0+//xztWnTRmPGjHH0ueeee/TLL79o+vTpqlq1ql588UU1aNBAS5YsKdA4PD09/7VPvXr1JJ2bE1TSXKx+Y0wxVwJc2QhGAPJUvXp15ebmas+ePU7tR44c0cmTJ1W9evVLbh8QEKB7771Xs2bN0sGDBxUTE6MJEyY4TRSuUqWKHnvsMS1YsED79+9XcHCwJkyY4Di+dO65P/+UnJz8r8fPS9euXSVJH3744b/2vdTxf/rpJ1111VUKCAhQ5cqVVa5cuQv+ni62bUGdr2fv3r1O7cePH3c6Kwbg8hCMAOSpS5cukqRp06Y5tb/88suSzs31uZjjx487ffbx8VFERISMMcrKylJOTo7S09Od+oSEhKhq1aqORwE0b95cISEhmjFjhtPjAZYsWaKkpKRLHv9iIiMj1alTJ7377rtasGDBBevPnj2rJ598UtK50Na0aVPNmTNHJ0+edPT58ccf9c033zj+fjw9PRUdHa0FCxbo4MGDjn5JSUlatmyZyzVeTIcOHeTl5aW33nrLqf31118vtGMA4HZ9ABfRpEkT9enTR++8845Onjyptm3bauPGjZozZ466d++uW2+99aLbduzYUWFhYbrpppsUGhqqpKQkvf7664qJiVGFChV08uRJXXPNNbr77rvVpEkTlS9fXitWrNCmTZv00ksvSZK8vb01efJk9evXT23bttX999/vuF2/Ro0aGjZsWIHG9cEHH6hjx46666671LVrV3Xo0EEBAQHas2eP5s6dq5SUFMezjF588UV17txZkZGR6t+/v+N2/cDAQI0dO9axz3Hjxmnp0qW6+eab9dhjjyk7O9vxHKcffvihQHX+U2hoqB5//HG99NJLuuOOO9SpUyft2LFDS5Ys0VVXXVWgS54A8uDu2+IAuMf528Q3bdp00T5ZWVlm3LhxpmbNmsbb29uEh4ebuLg4c+bMGad+/7xd/+233za33HKLCQ4ONr6+vqZWrVpm5MiRJj093RhjTGZmphk5cqRp0qSJqVChggkICDBNmjQxb7755gU1zJs3z1x//fXG19fXVKpUycTGxprffvvNqU+fPn1MQEDABduOGTPmgtvljTEmIyPDTJ061bRo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jkIrDgAAoDi5fLu+h8fFTzJZLBbl5ORcdlHFjdv1AQAofUrE7fr/vD0fAACgrHB5jhEAAEBZle8zRq+99lq++g0dOrTAxQAAALhTvucY1axZ8993ZrHol19+ueyiihtzjAAAKH3cOsdo//79hXJAAACAkoo5RgDKrJycDK1cVUsrV9VSTk6Gu8sBUArwqmkAZZanp786tN/n7jIAlCKcMQIAALAjGAEAANgRjAAAAOwKNMcoNzdXe/fu1dGjRy94EvYtt9xSKIUBAAAUN5eD0ffff6+ePXvqwIED+ucjkErru9IAAACkAlxKGzhwoJo3b64ff/xRaWlpOnHihGNJS0srcCGTJk2SxWLRE0884Wg7c+aMBg0apODgYJUvX149evTQkSNHnLY7ePCgYmJi5O/vr5CQEI0cOVLZ2dkFrgMAAFy5XD5jtGfPHs2fP1+1a9cutCI2bdqkt99+W40bN3ZqHzZsmBYvXqzPPvtMgYGBGjx4sO666y6tW7dOkpSTk6OYmBiFhYVp/fr1SklJUe/eveXt7a2JEycWWn0AAODK4PIZo1atWmnv3r2FVsCpU6cUGxurmTNnqmLFio729PR0vffee3r55ZfVvn17NWvWTLNmzdL69ev1/fffS5K++eYb7d69Wx9++KGaNm2qzp076/nnn9cbb7yhs2fPFlqNAADgyuByMBoyZIhGjBih2bNna8uWLfrhhx+cFlcNGjRIMTExioqKcmrfsmWLsrKynNrr1aunatWqKTExUZKUmJioRo0aKTQ01NEnOjpaNptNu3btuugxMzMzZbPZnBYAAACXL6X16NFDkvTggw862iwWi4wxLk++njt3rrZu3apNmzZdsC41NVU+Pj4KCgpyag8NDVVqaqqjz99D0fn159ddTHx8vMaNG5fvOgEAwJXB5WBUWC+TPXTokB5//HEtX75cfn5+hbLP/IqLi9Pw4cMdn202m8LDw4u1BgAAUPK4HIyqV69eKAfesmWLjh49qhtuuMHRlpOTo++++06vv/66li1bprNnz+rkyZNOZ42OHDmisLAwSVJYWJg2btzotN/zd62d75MXX19f+fr6Fso4AJRcWWfO6LU+d0uShs6ZL+9i/iUMQOlToAc87tu3T9OmTVNSUpIkKSIiQo8//rhq1aqV73106NBBO3fudGrr16+f6tWrp1GjRik8PFze3t5a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ANgRjCT5+vpqzJgx8vX1dXcpBVYWxiAxjpKkLIxBKhvjKAtjkBhHSVIWxiAVzTgspjDvcQMAACjFOGMEAABgRzACAACwIxgBAADYEYwAAADsrshglJaWptjYWFmtVgUFBal///46derUJbdp166dLBaL0zJw4MBiqjhvBRnHecYYde7cWRaLRQsWLCjaQv9FQcbxyCOPqFatWipXrpwqV66sbt266aeffiqmii/k6hjS0tI0ZMgQ1a1bV+XKlVO1atU0dOhQpaenF2PVedfl6s/inXfeUbt27WS1WmWxWHTy5MniKfZv3njjDdWoUUN+fn5q1aqVNm7ceMn+n332merVqyc/Pz81atRIX3/9dTFVenGujGHXrl3q0aOHatSoIYvFomnTphVfof/ClXHMnDlTN998sypWrKiKFSsqKirqX392xcWVcXzxxRdq3ry5goKCFBAQoKZNm+r//u//irHavLn67+K8uXPnymKxqHv37kVbYD65Mo7Zs2df8F3t5+fn2gHNFahTp06mSZMm5vvvvzf/+9//TO3atc39999/yW3atm1rHn74YZOSkuJY0tPTi6nivBVkHOe9/PLLpnPnzkaSSUhIKNpC/0VBxvH222+bNWvWmP3795stW7aYrl27mvDwcJOdnV1MVTtzdQw7d+40d911l1m4cKHZu3evWblypalTp47p0aNHMVZ9oYL8LF555RUTHx9v4uPjjSRz4sSJ4inWbu7cucbHx8e8//77ZteuXebhhx82QUFB5siRI3n2X7dunfH09DRTpkwxu3fvNs8884zx9vY2O3fuLNa6/87VMWzcuNE8+eST5pNPPjFhYWHmlVdeKd6CL8LVcfTs2dO88cYbZtu2bSYpKcn07dvXBAYGmt9++62YK3fm6ji+/fZb88UXX5jdu3ebvXv3mmnTphlPT0+zdOnSYq78/3N1DOft37/fXH311ebmm2823bp1K55iL8HVccyaNctYrVan7+rU1FSXjnnFBaPdu3cbSWbTpk2OtiVLlhiLxWJ+//33i27Xtm1b8/jjjxdDhflT0HEYY8y2bdvM1VdfbVJSUtwejC5nHH+3Y8cOI8ns3bu3KMq8pMIaw6effmp8fHxMVlZWUZT5ry53HN9++61bglHLli3NoEGDHJ9zcnJM1apVTXx8fJ7977nnHhMTE+PU1qpVK/PII48UaZ2X4uoY/q569eolJhhdzjiMMSY7O9tUqFDBzJkzp6hKzJfLHYcxxlx//fXmmWeeKYry8qUgY8jOzjatW7c27777runTp0+JCEaujmPWrFkmMDDwso55xV1KS0xMVFBQkJo3b+5oi4qKkoeHhzZs2HDJbT/66CNdddVVatiwoeLi4pSRkVHU5V5UQceRkZGhnj176o033lBYWFhxlHpJl/PzOO/06dOaNWuWatasqfDw8KIq9aIKYwzSuZchWq1WeXm55xWGhTWO4nT27Flt2bJFUVFRjjYPDw9FRUUpMTExz20SExOd+ktSdHT0RfsXtYKMoSQqjHFkZGQoKyurUF8I6qrLHYcxRitXrlRycrJuueWWoiz1ogo6hvHjxyskJET9+/cvjjL/VUHHcerUKVWvXl3h4eHq1q2bdu3a5dJxr7iXyKampiokJMSpzcvLS5UqVVJqaupFt+vZs6eqV6+uqlWr6ocfftCoUaOUnJysL774oqhLzlNBxzFs2DC1bt1a3bp1K+oS86Wg45CkN998U0899ZROnz6tunXravny5fLx8SnKcvN0OWM4748//tDzzz+vAQMGFEWJ+VIY4yhuf/zxh3JychQaGurUHhoaetE5Z6mpqXn2d9cYCzKGkqgwxjFq1ChVrVr1guBanAo6jvT0dF199dXKzMyUp6en3nzzTd12221FXW6eCjKGtWvX6r333tP27duLocL8Kcg46tatq/fff1+NGzdWenq6pk6dqtatW2vXrl265ppr8nXcMnPGaPTo0RdMuPrncjn/kxkwYICio6PVqFEjxcbG6oMPPlBCQoL27dtXiKMo2nEsXLhQq1atKpaJmkX985Ck2NhYbdu2TWvWrNF1112ne+65R2fOnCmkERTPGCTJZrMpJiZGERERGjt27OUX/g/FNQ7gckyaNElz585VQkKC65NlS4AKFSpo+/bt2rRpkyZMmKDhw4dr9erV7i4rX/7880/16tVLM2fO1FVXXeXuci5LZGSkevfuraZNm6pt27b64osvVLlyZb399tv53keZOWM0YsQI9e3b95J9rr32WoWFheno0aNO7dnZ2UpLS3Pp0lKrVq0kSXv37lWtWrVcrvdiinIcq1at0r59+xQUFOTU3qNHD918882F+o+4OH4egYGBCgwMVJ06dXTjjTeqYsWKSkhI0P3333+55UsqnjH8+eef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W5u4uDizY8cOpz7K44zRu+++61j/ySefGElm5cqVjrb4+HhTt25dx+eCnDH6p88++8wEBwc7Ps+aNcsEBgZe0C+vM0bPPPOM4/OpU6eMJLNkyRJjjDGjRo0yDRs2dNrH008/fckzRsYY89///td4eXmZSpUqmU6dOpkpU6aY1NRUx/rzZ37+vo8zZ84Yf39/s379eqd99e/f39x///1O2/39be3Hjx835cqVM/PmzbtoPQAKhjNGABx69Oihw4cPa+HCherUqZNWr16tG264QbNnz77kdo0bN3b8OTQ0VJLUqFEjp7bzk5ALasWKFerQoYOuvvpqVahQQb169dLx48eVkZHh8r7+Xm9AQICsVqujvuTkZLVo0cKpf8uWLf91nxMmTFBqaqpmzJihBg0aaMaMGapXr5527tx50W327t2rjIwM3XbbbY4zduXLl9cHH3ygffv2OfWNjIx0/LlSpUqqW7eukpKS8jVeAPlHMALgxM/PT7fddpueffZZrV+/Xn379tWYMWMuuY23t7fjzxaLJc+23Nxcx2cPD48LLjld6rLQr7/+qttvv12NGzfW559/ri1btuiNN96QJJ09ezb/g8uj3rzqK6jg4GD95z//0dSpU5WUlKSqVatq6tSpF+1/6tQpSdLixYu1fft2x7J7927Nnz//susB4DqCEYBLioiIKPQ7qypXrqyUlBTH55ycHP34448X7b9lyxbl5ubqpZde0o033qjrrrtOhw8fdurj4+OjnJycy66tbt262rx5s1Pbpk2bXN6Pj4+PatWq5fi78/HxkSSnGiMiIuTr66uDBw+qdu3aTkt4eLjT/r7//nvHn0+cOKGff/5Z9evXd7kuAJfG7foAJEnHjx/Xf/7zHz344INq3LixKlSooM2bN2vKlCnq1q1boR6rffv2Gj58uBYvXqxatWrp5ZdfvuTzfWrXrq2srCxNnz5dXbt21bp16zRjxgynPjVq1NCpU6e0cuVKNWnSRP7+/vL393e5tkceeUQvv/yyRo0apf79+2v79u2OS4nnz4b906JFizR37lzdd999uu6662SM0VdffaWvv/5as2bNkiRVr15dFotFixYtUpcuXVSuXDlVqFBBTz75pIYNG6bc3Fy1adNG6enpWrdunaxWq/r06eM4xvjx4xUcHKzQ0FA9/fTTuuqqq9S9e3eXxwfg0jhjBEDSubvSWrVqpVdeeUW33HKLGjZsqGeffVYPP/ywXn/99UI91oMPPqg+ffqod+/eatu2ra699lrdeuutF+3fpEkTvfzyy5o8ebIaNmyojz76SPHx8U59WrdurYEDB+ree+9V5cqVNWXKlALVVrNmTc2fP19ffPGFGjdurLfeestxV5qvr2+e20RERMjf318jRoxQ06ZNdeONN+rTTz/Vu+++q169ekmSrr76ao0bN06jR49WaGioBg8eLEl6/vnn9eyzzyo+Pl7169dXp06dtHjxYtWsWdPpGJMmTdLjjz+uZs2aKTU1VV999ZXjLBSAwmMx/7zQDwBwMmHCBM2YMUOHDh0q9mOvXr1at956q06cOJHnk70BFC4upQHAP7z55ptq0aKFgoODtW7dOr344ouOMzwAyjaCEQD8w549e/TCCy8oLS1N1apV04gRIxQXF+fusgAUAy6lAQAA2DH5GgAAwI5gBAAAYEcwAgAAsCMYAQAA2BGMAAAA7AhGAAAAdgQjAAAAO4IRAACAHcEIAADA7v8BUdAD48VoccgAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x12ffeb880>"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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hautrXWfffaZKyoqcunp6e7KlSvWo8VVS0uLO3v2rDt79qyT5N5++2139uxZ99VXXznnnPvDH/7gUlNT3aFDh9z58+fdwoULXV5enrt586bx5LH1oPPQ0tLiXnvtNVdVVeVqa2vdxx9/7J577jk3btw419bWZj16zKxbt875fD5XUVHhGhsbQ+vGjRuhY9auXetyc3PdsWPH3KlTp1xBQYErKCgwnDr2HnYeampq3G9/+1t36tQpV1tb6w4dOuRGjx7tCgsLjScP1ycC5Jxzf/nLX1xubq4bPHiwmzFjhjtx4oT1SD1u2bJlLisryw0ePNg9/fTTbtmyZa6mpsZ6rLj75JNPnKT71ooVK5xzd96K/cYbb7jMzEzn9XrdnDlzXHV1te3QcfCg83Djxg03d+5cN2LECDdo0CA3atQot3r16oT7j7Su/vdLcjt37gwdc/PmTfezn/3MPfXUU27YsGFu8eLFrrGx0W7oOHjYeairq3OFhYUuLS3Neb1eN3bsWPf666+7QCBgO/g9+OsYAAAmev1rQACAxESAAAAmCBAAwAQBAgCYIEAAABMECABgggABAEwQIACACQIEADBBgAAAJggQAMAEAQIAmPh/Cyk6yIXeoA0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualise a random image from the test datset\n",
    "img_batch, label_batch = next(iter(test_loader))\n",
    "\n",
    "sample_ind = np.random.randint(args['test_batch_size'])\n",
    "\n",
    "sample_img = img_batch[sample_ind]\n",
    "x = torch.flatten(sample_img)\n",
    "N = x.shape[0]\n",
    "out_spike = torch.zeros([args['T'], N])\n",
    "\n",
    "\n",
    "for t in range(args['T']):\n",
    "    out_spike[t] = encoder_sample(x)\n",
    "\n",
    "visualizing.plot_1d_spikes(out_spike.numpy(), 'PoissonCoding', 'Simulating Step', 'Neuron Index',\n",
    "                           plot_firing_rate=False)\n",
    "plt.show()\n",
    "plt.imshow(sample_img.squeeze(axis=0), cmap='gray')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "1ddac928",
   "metadata": {},
   "outputs": [],
   "source": [
    "def thresholded_image_encoding_device(X: torch.Tensor, threshold: torch.Tensor):\n",
    "    # Move threshold to the same device as X\n",
    "    threshold = threshold.to(X.device)\n",
    "\n",
    "    # Check if threshold is a scalar tensor\n",
    "    if threshold.numel() == 1:  # Check if threshold has only one element\n",
    "        # Expand the scalar threshold to match the batch size\n",
    "        threshold = threshold.expand(X.shape[0])  # Replicate threshold for each batch\n",
    "\n",
    "    # Check if threshold size matches batch size\n",
    "    if threshold.shape[0] != X.shape[0]:\n",
    "        print(\"Threshold size does not match the batch-size\")\n",
    "        raise ValueError(\"Threshold size does not match the batch-size\")\n",
    "\n",
    "    # Expand the threshold to match the dimensions of X for broadcasting\n",
    "    threshold_expanded = threshold.view(-1, 1, 1, 1)\n",
    "\n",
    "    # Apply the threshold to X for each batch\n",
    "    thresholded_image = torch.where(X > threshold_expanded, 1, 0)\n",
    "\n",
    "    return thresholded_image.float()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d418e78a",
   "metadata": {
    "id": "d418e78a"
   },
   "source": [
    "# Define the SNN models"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "757c925b",
   "metadata": {
    "id": "757c925b"
   },
   "source": [
    "### Layer with {0, 1} weights using STE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "d68c6946",
   "metadata": {
    "id": "d68c6946"
   },
   "outputs": [],
   "source": [
    "# Define Binary Step function \n",
    "class BinaryStepFunction_0_1(torch.autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, input):\n",
    "        return (input.sign() + 1)/2\n",
    "    \n",
    "    # STE\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_output):\n",
    "        return grad_output.clone()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "658b26e5",
   "metadata": {
    "id": "658b26e5"
   },
   "outputs": [],
   "source": [
    "class BinaryLayer_0_1(nn.Module):\n",
    "    def __init__(self, input_features, output_features):\n",
    "        super(BinaryLayer_0_1, self).__init__()\n",
    "        self.weight = nn.Parameter(torch.Tensor(output_features, input_features))\n",
    "        nn.init.kaiming_uniform_(self.weight, a=math.sqrt(5))  # Weight initialization\n",
    "\n",
    "    def forward(self, x):\n",
    "        binary_weight = BinaryStepFunction_0_1.apply(self.weight - args['epsilon'])\n",
    "        return F.linear(x, binary_weight)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "974fd922",
   "metadata": {},
   "outputs": [],
   "source": [
    "class SNN_model_thLR(nn.Module):\n",
    "    def __init__(self, in_channels, out_channels, hidden_neurons, init_th_1=1.0, init_th_2=1.0):  \n",
    "        super(SNN_model_thLR, self).__init__()\n",
    "\n",
    "        # First layer and neuron\n",
    "        self.layer1 = BinaryLayer_0_1(in_channels, hidden_neurons)\n",
    "        self.v_threshold1 = nn.Parameter(torch.tensor( init_th_1 ))  # Learnable threshold for neuron1\n",
    "        self.neuron1 = neuron.IFNode(surrogate_function=surrogate.ATan())\n",
    "\n",
    "        # Second layer and neuron\n",
    "        self.layer2 = BinaryLayer_0_1(hidden_neurons, out_channels)\n",
    "        self.v_threshold2 = nn.Parameter(torch.tensor( init_th_2 ))  # Learnable threshold for neuron2\n",
    "        self.neuron2 = neuron.IFNode(surrogate_function=surrogate.ATan())\n",
    "\n",
    "\n",
    "        # Store the emitted spike train from different layers\n",
    "        self.y_1 = []\n",
    "        self.y_2 = []\n",
    "\n",
    "    def forward(self, x):\n",
    "        # Pass through first layer\n",
    "        y = self.layer1(x)\n",
    "       \n",
    "        self.neuron1.v_threshold = self.v_threshold1 \n",
    "        y = self.neuron1(y)\n",
    "\n",
    "        self.y_1 = y\n",
    "\n",
    "        # Pass through second layer\n",
    "        y = self.layer2(y)\n",
    "        \n",
    "        self.neuron2.v_threshold = self.v_threshold2  \n",
    "        y = self.neuron2(y)\n",
    "\n",
    "        self.y_2 = y\n",
    "\n",
    "\n",
    "        return y"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0314fcd9",
   "metadata": {
    "id": "0314fcd9"
   },
   "source": [
    "### Layer with {-1, 0, 1} weights using STE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "64bd84aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "class BinaryStepFunction_ternary(torch.autograd.Function):\n",
    "\n",
    "    @staticmethod\n",
    "    def forward(ctx, input):\n",
    "        return input.sign()\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_output):\n",
    "        return grad_output.clone()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "3642eb65",
   "metadata": {},
   "outputs": [],
   "source": [
    "class TernaryLayer(nn.Module):\n",
    "    def __init__(self, input_features, output_features):\n",
    "        super(TernaryLayer, self).__init__()\n",
    "        self.weight = nn.Parameter(torch.Tensor(output_features, input_features))\n",
    "        nn.init.kaiming_uniform_(self.weight, a=math.sqrt(5))  # Weight initialization\n",
    "\n",
    "    def forward(self, x):\n",
    "        ternary_weight = BinaryStepFunction_ternary.apply(self.weight - args['epsilon'])\n",
    "        return F.linear(x, ternary_weight)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "147bda0c",
   "metadata": {},
   "outputs": [],
   "source": [
    "class SNN_model_ternary_thLR(nn.Module):\n",
    "    def __init__(self, in_channels, out_channels, hidden_neurons, init_th_1=1.0, init_th_2=1.0):  \n",
    "        super(SNN_model_ternary_thLR, self).__init__()\n",
    "\n",
    "        # First layer neuron\n",
    "        self.layer1 = TernaryLayer(in_channels, hidden_neurons)\n",
    "        self.v_threshold1 = nn.Parameter(torch.tensor( init_th_1 ))  # Learnable threshold for neuron1\n",
    "        self.neuron1 = neuron.IFNode(surrogate_function=surrogate.ATan())\n",
    "\n",
    "        # Second layer neuron\n",
    "        self.layer2 = TernaryLayer(hidden_neurons, out_channels)\n",
    "        self.v_threshold2 = nn.Parameter(torch.tensor( init_th_2 ))  # Learnable threshold for neuron2\n",
    "        self.neuron2 = neuron.IFNode(surrogate_function=surrogate.ATan())\n",
    "\n",
    "    def forward(self, x):\n",
    "        # Pass through first layer\n",
    "        y = self.layer1(x)\n",
    "       \n",
    "        self.neuron1.v_threshold = self.v_threshold1 \n",
    "        y = self.neuron1(y)\n",
    "\n",
    "\n",
    "        # Pass through second layer\n",
    "        y = self.layer2(y)\n",
    "        \n",
    "        self.neuron2.v_threshold = self.v_threshold2  \n",
    "        y = self.neuron2(y)\n",
    "\n",
    "        return y\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "815caf39",
   "metadata": {},
   "source": [
    "# Define the threshold binarised image encoding "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "a22d6970",
   "metadata": {},
   "outputs": [],
   "source": [
    "def thresholded_image_encoding_device(X: torch.Tensor, threshold: torch.Tensor):\n",
    "    # Move threshold to the same device as X\n",
    "    threshold = threshold.to(X.device)\n",
    "\n",
    "    # Check if threshold is a scalar tensor\n",
    "    if threshold.numel() == 1:  # Check if threshold has only one element\n",
    "        # Expand the scalar threshold to match the batch size\n",
    "        threshold = threshold.expand(X.shape[0])  # Replicate threshold for each batch\n",
    "\n",
    "    # Check if threshold size matches batch size\n",
    "    if threshold.shape[0] != X.shape[0]:\n",
    "        print(\"Threshold size does not match the batch-size\")\n",
    "        raise ValueError(\"Threshold size does not match the batch-size\")\n",
    "\n",
    "    # Expand the threshold to match the dimensions of X for broadcasting\n",
    "    threshold_expanded = threshold.view(-1, 1, 1, 1)\n",
    "\n",
    "    # Apply the threshold to X for each batch\n",
    "    thresholded_image = torch.where(X > threshold_expanded, 1, 0)\n",
    "\n",
    "    return thresholded_image.float()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ce4d00fe",
   "metadata": {
    "id": "ce4d00fe"
   },
   "source": [
    "# Load pre-trained weights to extract explanation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9ffa5121",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "9ffa5121",
    "outputId": "27eabee5-53e3-48d1-e6ae-91d73ec34fb1"
   },
   "outputs": [],
   "source": [
    "# Load best model weights from checkpoint file\n",
    "weights_checkpoint = 'final_exps/3-digit/chckpt_bin_BIN-ENC_=_(3.5348103046417236, 1.0024094581604004)_classes=[1, 5, 9]_numNeur=8_epoch_37.pth'\n",
    "\n",
    "best_snnNet = SNN_model_thLR(in_channels=28*28, out_channels=args['num_classes'], hidden_neurons=args['hidden_neurons'])\n",
    "\n",
    "best_snnNet.to('cpu')\n",
    "# Load the state dict\n",
    "state_dict = torch.load(weights_checkpoint, map_location='cpu')\n",
    "\n",
    "# Remove the unnecessary neuron-specific v_threshold entries\n",
    "state_dict.pop(\"neuron1.v_threshold\", None)\n",
    "state_dict.pop(\"neuron2.v_threshold\", None)\n",
    "\n",
    "# Load the modified state dict into the model\n",
    "best_snnNet.load_state_dict(state_dict)\n",
    "best_snnNet.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "c59df284",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy: 93.4138%\n"
     ]
    }
   ],
   "source": [
    "# TEST SET ACCURACY\n",
    "\n",
    "encoder_test = encoding.PoissonEncoder()\n",
    "best_snnNet.eval()\n",
    "\n",
    "test_samples = 0.\n",
    "test_acc = 0.\n",
    "\n",
    "with torch.no_grad():\n",
    "\n",
    "    for test_batch_idx, (img, label) in enumerate(test_loader):\n",
    "        \n",
    "        label.apply_(label_map.get)\n",
    "        \n",
    "        img = img.to('cpu')\n",
    "        label = label.to('cpu')\n",
    "\n",
    "        out_fr = 0.\n",
    "\n",
    "        for t in range(args['T']):\n",
    "            encoded_img = encoder_test(img) #thresholded_image_encoding_device(img, torch.tensor([0.5], dtype=torch.float32))\n",
    "            x = torch.flatten(encoded_img, start_dim=1)\n",
    "            out_fr += best_snnNet(x)\n",
    "\n",
    "        out_fr = out_fr / args['T']\n",
    "\n",
    "        test_samples += label.numel()\n",
    "        # The correct rate is calculated as follows. The subscript i of the neuron with the highest firing rate in the output layer is considered as the result of classification.\n",
    "        test_acc += (out_fr.argmax(1) == label).float().sum().item()\n",
    "        \n",
    "        # After optimizing the parameters, the state of the network should be reset because the neurons of the SNN have “memory”.\n",
    "        functional.reset_net(best_snnNet)\n",
    "\n",
    "print(\"Test accuracy: {:.4f}%\".format(test_acc/test_samples*100))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "bd4ef584",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation accuracy: 91.2941%\n"
     ]
    }
   ],
   "source": [
    "# VALIDATION SET ACCURACY\n",
    "\n",
    "encoder_val = encoding.PoissonEncoder()\n",
    "\n",
    "best_snnNet.eval()\n",
    "\n",
    "val_samples = 0.\n",
    "val_acc = 0.\n",
    "\n",
    "with torch.no_grad():\n",
    "\n",
    "    for val_batch_idx, (img, label) in enumerate(val_loader):\n",
    "        \n",
    "        label.apply_(label_map.get)\n",
    "        \n",
    "        img = img.to('cpu')\n",
    "        label = label.to('cpu')\n",
    "\n",
    "        out_fr = 0.\n",
    "\n",
    "        for t in range(args['T']):\n",
    "            encoded_img =  encoder_val(img) # thresholded_image_encoding_device(img, torch.tensor([0.5], dtype=torch.float32))\n",
    "            x = torch.flatten(encoded_img, start_dim=1)\n",
    "            out_fr += best_snnNet(x)\n",
    "\n",
    "        out_fr = out_fr / args['T']\n",
    "\n",
    "        val_samples += label.numel()\n",
    "        # The correct rate is calculated as follows. The subscript i of the neuron with the highest firing rate in the output layer is considered as the result of classification.\n",
    "        val_acc += (out_fr.argmax(1) == label).float().sum().item()\n",
    "        \n",
    "        # After optimizing the parameters, the state of the network should be reset because the neurons of the SNN have “memory”.\n",
    "        functional.reset_net(best_snnNet)\n",
    "        \n",
    "print(\"Validation accuracy: {:.4f}%\".format(val_acc/val_samples*100))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f759c952",
   "metadata": {
    "id": "f759c952"
   },
   "source": [
    "### Define BCM class "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 142,
   "id": "5f0d7240",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "501fbd68",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the Binary Causal Model (BCM) class that takes a pre-trained BSNN and as test dataset as input to convert to the appropriate causal model \n",
    "class BCM():\n",
    "    \n",
    "    def __init__(self, snnModel):\n",
    "            super(BCM, self).__init__()\n",
    "            \n",
    "            # Set the pretrained BSNN model and the data loader for loading images in batch \n",
    "            self.snnModel = snnModel\n",
    "            \n",
    "            # Initialise dictionary to store the incoming spikes at every time-step\n",
    "            self.input_spike_dict = {t: [] for t in range(args[\"T\"]) } # No input at t = -1 # Dim = T X N , N = |Test dataset| \n",
    "\n",
    "            # Initialise dictionaries to store the output spikes for each layer at every time-step\n",
    "            self.activation_dict_layer1 = {t : torch.zeros(len(test_dataset), args['hidden_neurons']) if t == -1 else [] for t in range(-1,args[\"T\"]) } \n",
    "            self.spike_dict_layer1 = {t : torch.zeros(len(test_dataset), args['hidden_neurons']) if t == -1 else  [] for t in range(-1,args[\"T\"]) }\n",
    "\n",
    "            self.activation_dict_layer2 = {t : torch.zeros(len(test_dataset), args['num_classes']) if t == -1 else  [] for t in range(-1,args[\"T\"]) }\n",
    "            self.spike_dict_layer2 = {t : torch.zeros(len(test_dataset), args['num_classes']) if t == -1 else  [] for t in range(-1,args[\"T\"]) }\n",
    "            \n",
    "            # Store the ground truth and prediction for all images\n",
    "            self.ground = []\n",
    "            self.pred = []\n",
    "            \n",
    "            # Construct the BCM vertices\n",
    "            self.nodes = self.endoExoVar()\n",
    "            \n",
    "            # Construct the BCM edges (R^+)\n",
    "            self.nonzero_connections_l1, self.nonzero_connections_l2 = self.extractNZ(args['epsilon'])\n",
    "                        \n",
    "            \n",
    "    # Define the dictionary of nodes in the BCM (U_s, V_s)       \n",
    "    def endoExoVar(self):\n",
    "        \n",
    "        # Dictionary to hold the exogeneous and endogeneous variables in the BCM \n",
    "        nodes = {'exo' : {t: [] for t in range(-1,args['T'])}, 'endo' : {'layer1' : {t: [] for t in range(-1, args['T'])}, 'layer2' : {t: [] for t in range(-1, args['T'])} }}\n",
    "\n",
    "        for t in range(-1,args['T']):\n",
    "            for ind in range(args['input-size']):\n",
    "                nodes['exo'][t].append(f'u_{ind},{t}')\n",
    "\n",
    "\n",
    "            for ind in range(args['hidden_neurons']):\n",
    "                nodes['endo']['layer1'][t].append(f'v1_{ind},{t}')\n",
    "\n",
    "            for ind in range(args['num_classes']):\n",
    "                nodes['endo']['layer2'][t].append(f'v2_{ind},{t}')\n",
    "        return nodes   \n",
    "    \n",
    "    # Evaluate the pre-trained SNN model on test images for creating instances of BCM\n",
    "    def evalSNN(self, test_loader):\n",
    "\n",
    "            # Use a Poisson Encoder for encoding test images\n",
    "            encoder_test = encoding.PoissonEncoder()\n",
    "\n",
    "            self.snnModel.eval()\n",
    "\n",
    "            with torch.no_grad():\n",
    "\n",
    "                for test_batch_idx, (img, label) in enumerate(test_loader):\n",
    "\n",
    "                    self.ground.extend(label.clone().tolist())\n",
    "                    label.apply_(label_map.get)\n",
    "\n",
    "                    img = img.to('cpu')\n",
    "                    label = label.to('cpu')\n",
    "\n",
    "                    out_fr = 0.\n",
    "\n",
    "                    for t in range(args['T']):\n",
    "                        encoded_img = encoder_test(img) #thresholded_image_encoding_device(img, torch.tensor([0.5], dtype=torch.float32))\n",
    "\n",
    "                        x = torch.flatten(encoded_img, start_dim=1)\n",
    "                        self.input_spike_dict[t].extend(x.tolist())\n",
    "                        out_fr += self.snnModel(x)\n",
    "\n",
    "                        self.activation_dict_layer1[t].extend(self.snnModel.neuron1.v.tolist())\n",
    "                        self.activation_dict_layer2[t].extend(self.snnModel.neuron2.v.tolist())\n",
    "\n",
    "                        self.spike_dict_layer1[t].extend(self.snnModel.y_1.tolist())\n",
    "                        self.spike_dict_layer2[t].extend(self.snnModel.y_2.tolist())\n",
    "\n",
    "                    out_fr = out_fr / args['T']\n",
    "\n",
    "                    self.pred.extend(out_fr.argmax(1).apply_(label_map_inverse.get).tolist())\n",
    "\n",
    "                    # After optimizing the parameters, the state of the network is reset\n",
    "                    functional.reset_net(self.snnModel)\n",
    "                    \n",
    "                    \n",
    "                    \n",
    "    # Extract the addresses of neurons with non-zero connections (R^+)\n",
    "    def extractNZ(self, epsilon):\n",
    "        \n",
    "        # Apply binarising function on the full precision weights\n",
    "        for par_ind, param in enumerate(self.snnModel.parameters()):\n",
    "            if par_ind == 2:\n",
    "                binary_param_l1 = BinaryStepFunction_0_1.apply(param - epsilon)\n",
    "                \n",
    "            elif par_ind == 3:\n",
    "                binary_param_l2 = BinaryStepFunction_0_1.apply(param - epsilon)\n",
    "\n",
    "        \n",
    "        nonzero_connections_l1 = []\n",
    "        input_range_l1 = torch.arange(args['input-size'])\n",
    "        for ind in range(args['hidden_neurons']):\n",
    "            nonzero_connections_l1.append(input_range_l1[(binary_param_l1 == 1)[ind]].tolist())\n",
    "\n",
    "\n",
    "        nonzero_connections_l2 = []\n",
    "        input_range_l2 = torch.arange(args['hidden_neurons'])\n",
    "        for ind in range(args['num_classes']):\n",
    "            nonzero_connections_l2.append(input_range_l2[(binary_param_l2 == 1)[ind]].tolist())\n",
    "            \n",
    "            \n",
    "        return [nonzero_connections_l1, nonzero_connections_l2]\n",
    "    \n",
    "    def structEqu(self, img_ind, time_slice):\n",
    "\n",
    "        start_time = time.time()\n",
    "\n",
    "        # Define a dictionary to store the structural equations for each layer and time-step\n",
    "        equations = {'layer1': {}, 'layer2': {}}\n",
    "\n",
    "        v_thresh_1 = self.snnModel.v_threshold1.item().__ceil__()\n",
    "        v_thresh_2 = self.snnModel.v_threshold2.item().__ceil__()\n",
    "\n",
    "        # Precompute and store reusable parts\n",
    "        nonzero_l1_connections = self.nonzero_connections_l1\n",
    "        nonzero_l2_connections = self.nonzero_connections_l2\n",
    "        activation_dict_layer1 = self.activation_dict_layer1\n",
    "        activation_dict_layer2 = self.activation_dict_layer2\n",
    "\n",
    "        equations['layer1'][time_slice] = {}\n",
    "        equations['layer2'][time_slice] = {}\n",
    "\n",
    "        for ind in range(args['hidden_neurons']):\n",
    "            print(\"FInding structural equations for Layer 1, neuron id : {}\".format(ind))\n",
    "\n",
    "            prev_activation_l1 = activation_dict_layer1[time_slice-1][img_ind][ind]\n",
    "            combin_spike_l1 = combinations(nonzero_l1_connections[ind], v_thresh_1)\n",
    "            combin_noSpike_l1 = combinations(nonzero_l1_connections[ind], max(v_thresh_1 - int(prev_activation_l1), 0))\n",
    "\n",
    "            # Use generator expressions for lazy evaluation\n",
    "            DNF_spike_l1 = Or(*(And(*[Bool(f'u_{i},{time_slice}') for i in arr]) for arr in combin_spike_l1))\n",
    "            DNF_noSpike_l1 = Or(*(And(*[Bool(f'u_{i},{time_slice}') for i in arr]) for arr in combin_noSpike_l1))\n",
    "\n",
    "            equations['layer1'][time_slice][f'v1_{ind},{time_slice}'] = And(\n",
    "                Or(Bool(f'v1_{ind},{time_slice-1}'), DNF_noSpike_l1),\n",
    "                Or(Not(Bool(f'v1_{ind},{time_slice-1}')), DNF_spike_l1)\n",
    "            )\n",
    "\n",
    "        for ind in range(args['num_classes']):\n",
    "\n",
    "            print(\"FInding structural equations for Layer 2, neuron id : {}\".format(ind))\n",
    "\n",
    "            prev_activation_l2 = activation_dict_layer2[time_slice-1][img_ind][ind]\n",
    "            combin_spike_l2 = combinations(nonzero_l2_connections[ind], v_thresh_2)\n",
    "            combin_noSpike_l2 = combinations(nonzero_l2_connections[ind], max(v_thresh_2 - int(prev_activation_l2), 0))\n",
    "\n",
    "            DNF_spike_l2 = Or(*(And(*[Bool(f'v1_{i},{time_slice}') for i in arr]) for arr in combin_spike_l2))\n",
    "            DNF_noSpike_l2 = Or(*(And(*[Bool(f'v1_{i},{time_slice}') for i in arr]) for arr in combin_noSpike_l2))\n",
    "\n",
    "            equations['layer2'][time_slice][f'v2_{ind},{time_slice}'] = And(\n",
    "                Or(Bool(f'v2_{ind},{time_slice-1}'), DNF_noSpike_l2),\n",
    "                Or(Not(Bool(f'v2_{ind},{time_slice-1}')), DNF_spike_l2)\n",
    "            )\n",
    "\n",
    "            end_time = time.time()\n",
    "\n",
    "            print(\"Time to construct equations: {}\".format(end_time-start_time))\n",
    "\n",
    "        return equations\n",
    "    \n",
    "    \n",
    "        \n",
    "    def getEquSlice(self, time_slice, img_ind):\n",
    "            \n",
    "            # Add the causal model equations constraint for all endogeneous variables at the given time slice\n",
    "            equations = self.structEqu(img_ind, time_slice) # Set of all structural equations\n",
    "            equArr = []\n",
    "            \n",
    "            for var in equations['layer1'][time_slice].keys():\n",
    "                equArr.append(Bool(var) == equations['layer1'][time_slice][var])\n",
    "\n",
    "            for var in equations['layer2'][time_slice].keys():\n",
    "                equArr.append(Bool(var) == equations['layer2'][time_slice][var])\n",
    "                \n",
    "                \n",
    "            return equArr\n",
    "            \n",
    "    def computeAXP(self, img_ind, time_slice, verbose = 1):\n",
    "        \n",
    "        # Choose a time-slice to compute abductive explanation on the chosen image (img_ind)\n",
    "        input_time_slice = self.input_spike_dict[time_slice][img_ind]\n",
    "        out_spike_slice = self.spike_dict_layer2[time_slice][img_ind]\n",
    "        \n",
    "        # Add the initialisation constraint at time slice = -1\n",
    "        initConst = [Not(Bool(f'u_{i},{-1}')) for i in range(args['input-size']) ]\n",
    "        initConst.extend([Not(Bool(f'v1_{i},{-1}')) for i in range(args['hidden_neurons']) ])\n",
    "        initConst.extend([Not(Bool(f'v2_{i},{-1}')) for i in range(args['num_classes']) ])\n",
    "        \n",
    "        # Add the causal model equations constraint for all endogeneous variables at the given time slice\n",
    "        equArr = self.getEquSlice(time_slice, img_ind)\n",
    "            \n",
    "            \n",
    "        # Set up the explanans and explanandum according to the actual case\n",
    "\n",
    "        explanandum = And(*[ Bool(f'v2_{i},{time_slice}') if bool(out_spike_slice[i]) else Not(Bool(f'v2_{i},{time_slice}')) for i in range(len(out_spike_slice)) ])\n",
    "        lamb = self.getLambda(time_slice, img_ind)\n",
    "        \n",
    "        maxSize_Cube = len(lamb)\n",
    "        arr_cubes_final = []\n",
    "\n",
    "        abdXP = lamb[:]\n",
    "        \n",
    "        if verbose != 0 :\n",
    "            print(\"---------------------\")\n",
    "            print(\"^^^^^^^^^^^^^^^^^^^^^\")\n",
    "            print(\"Checking for validity\")\n",
    "            print(\"^^^^^^^^^^^^^^^^^^^^^\")\n",
    "            print(\"---------------------\")\n",
    "            print(\"\\n\")\n",
    "\n",
    "        start_time = time.time()\n",
    "        \n",
    "        \n",
    "        # RUN ALGO 1 to compute abductive explanation in pixel space\n",
    "        \n",
    "        for lit_id, literal in enumerate(lamb):\n",
    "\n",
    "           \n",
    "            \n",
    "            if verbose != 0:\n",
    "                print(\"------------------------------------\")\n",
    "                print(\" Searching Literal: {} with id: {}  \".format(literal, lit_id))\n",
    "                \n",
    "            tmp2 = abdXP[:]\n",
    "            tmp2.remove(literal)\n",
    "            \n",
    "            if verbose != 0:\n",
    "                print(' Cube size : {}'.format(len(tmp2)))\n",
    "\n",
    "            explanant = And(*tmp2)\n",
    "\n",
    "            solver = Solver()\n",
    "            solver.add(And(*initConst))\n",
    "\n",
    "            beta1 = Or(Not(And(And(*equArr), explanant)), explanandum)\n",
    "\n",
    "            solver.add(Not(beta1))\n",
    "            if verbose != 0:\n",
    "                print(\" {} \".format(solver.check()))\n",
    "\n",
    "            if repr(solver.check()) == 'unsat':\n",
    "\n",
    "                abdXP = tmp2\n",
    "                if verbose != 0:\n",
    "                    print(' ******************* ')\n",
    "                    print(\" Literal was removed \")\n",
    "                    print(' ******************* ')\n",
    "                    print(\"------------------------------------\")\n",
    "                    print('\\n')\n",
    "            else:\n",
    "                if verbose != 0:\n",
    "                    print(\" Literal was not removed \")\n",
    "                    print(\"------------------------------------\")\n",
    "                    print('\\n')\n",
    "\n",
    "        end_time = time.time()\n",
    "        \n",
    "        if verbose != 0:\n",
    "            print('Total search time: {:.4f}s'.format(end_time-start_time))\n",
    "            print('\\n')\n",
    "            print('\\n')\n",
    "        \n",
    "        return abdXP, end_time-start_time\n",
    "    \n",
    "    \n",
    "            \n",
    "    def getLambda(self, time_slice, img_ind, size = 'small'):\n",
    "        \n",
    "        \n",
    "        input_time_slice = self.input_spike_dict[time_slice][img_ind]\n",
    "        \n",
    "        # Flatten the address list of input neurons having nonzero connections with layer 1 neurons\n",
    "        nonzero_connections_flattened_l1 = []\n",
    "        for arr in self.nonzero_connections_l1:\n",
    "            nonzero_connections_flattened_l1.extend(arr)\n",
    "\n",
    "        nonzero_connections_set_l1 =  set(nonzero_connections_flattened_l1)\n",
    "        \n",
    "        \n",
    "        # Find input neurons having nonzero connections with those layer 1 neurons that have non-zero connection with layer 2 neurons\n",
    "        nonzero_connections_flattened_l2 = []\n",
    "        for arr in self.nonzero_connections_l2:\n",
    "            nonzero_connections_flattened_l2.extend(arr)\n",
    "\n",
    "        nonzero_connections_list_unique_l2 =  list(set(nonzero_connections_flattened_l2))\n",
    "\n",
    "        nonzero_connections_list_l1andl2 = []\n",
    "\n",
    "        for ind in nonzero_connections_list_unique_l2:\n",
    "            nonzero_connections_list_l1andl2.extend(self.nonzero_connections_l1[ind])\n",
    "\n",
    "        nonzero_connections_set_l1andl2 = set(nonzero_connections_list_l1andl2)\n",
    "        \n",
    "        lamb_big = [Bool(f'u_{i},{time_slice}') if bool(input_time_slice[i]) else Not(Bool(f'u_{i},{time_slice}'))  for i in range(args['input-size'])]\n",
    "        lamb_small = [Bool(f'u_{i},{time_slice}') if bool(input_time_slice[i]) else Not(Bool(f'u_{i},{time_slice}'))  for i in list(nonzero_connections_set_l1andl2)]\n",
    "        \n",
    "        if size == 'small':\n",
    "            lamb = lamb_small\n",
    "            \n",
    "        elif size == 'big':\n",
    "            lamb = lamb_big\n",
    "            \n",
    "        else:\n",
    "            print(\"Choose 'big' or 'small'\")\n",
    "            \n",
    "            \n",
    "        print(\"Initial guess implicant of length: {}\".format(len(lamb)))\n",
    "        \n",
    "        return lamb"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "36926f03",
   "metadata": {},
   "outputs": [],
   "source": [
    "bestSNN_BCM = BCM(best_snnNet)\n",
    "bestSNN_BCM.evalSNN(test_loader)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "b728ef68",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "5"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "args['img_ind'] = 115 + 6 # Change image to compute explanation\n",
    "args['time_slice'] = 0 # Change time slice to compute explanation\n",
    "\n",
    "bestSNN_BCM.ground[args['img_ind']]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "3c803a53",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Spiking values at time 0: Layer1 -> [0.0, 0.0, 1.0, 0.0, 1.0, 0.0, 0.0, 1.0], Layer2 -> [0.0, 1.0, 0.0]\n"
     ]
    }
   ],
   "source": [
    "for t in range(args['T']):\n",
    "    print(\"Spiking values at time {}: Layer1 -> {}, Layer2 -> {}\".format(t, bestSNN_BCM.spike_dict_layer1[t][args['img_ind']], bestSNN_BCM.spike_dict_layer2[t][args['img_ind']]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "d33382a5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Ground: 5 Prediction 5\n"
     ]
    }
   ],
   "source": [
    "plt.figure(figsize=(3,3))\n",
    "\n",
    "\n",
    "image_pixels, _ = test_dataset[args['img_ind']]\n",
    "\n",
    "image_timeslice = image_pixels.reshape(28, 28)\n",
    "plt.title(\"Image shown\\n at time step {}\".format(args['time_slice']))\n",
    "plt.imshow(image_timeslice, cmap='gray')\n",
    "\n",
    "plt.show()\n",
    "print(\"Ground: {} Prediction {}\".format(bestSNN_BCM.ground[args['img_ind']], bestSNN_BCM.pred[args['img_ind']]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "af465554",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initial guess implicant of length: 279\n",
      "---------------------\n",
      "^^^^^^^^^^^^^^^^^^^^^\n",
      "Checking for validity\n",
      "^^^^^^^^^^^^^^^^^^^^^\n",
      "---------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_252,2) with id: 0  \n",
      " Cube size : 278\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_15,2) with id: 1  \n",
      " Cube size : 278\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_115,2) with id: 2  \n",
      " Cube size : 278\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_769,2) with id: 3  \n",
      " Cube size : 278\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_702,2) with id: 4  \n",
      " Cube size : 277\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_304,2) with id: 5  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_729,2) with id: 6  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_34,2) with id: 7  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_757,2) with id: 8  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_22,2) with id: 9  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_64,2) with id: 10  \n",
      " Cube size : 276\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_747,2) with id: 11  \n",
      " Cube size : 276\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_5,2) with id: 12  \n",
      " Cube size : 275\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_54,2) with id: 13  \n",
      " Cube size : 274\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_248,2) with id: 14  \n",
      " Cube size : 273\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_13,2) with id: 15  \n",
      " Cube size : 273\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_715,2) with id: 16  \n",
      " Cube size : 273\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_644,2) with id: 17  \n",
      " Cube size : 272\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_140,2) with id: 18  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_196,2) with id: 19  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_76,2) with id: 20  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_759,2) with id: 21  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_525,2) with id: 22  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_551,2) with id: 23  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_616,2) with id: 24  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_65,2) with id: 25  \n",
      " Cube size : 271\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_28,2) with id: 26  \n",
      " Cube size : 271\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_537,2) with id: 27  \n",
      " Cube size : 270\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_83,2) with id: 28  \n",
      " Cube size : 270\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_618,2) with id: 29  \n",
      " Cube size : 270\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_87,2) with id: 30  \n",
      " Cube size : 269\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_676,2) with id: 31  \n",
      " Cube size : 269\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_111,2) with id: 32  \n",
      " Cube size : 268\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_699,2) with id: 33  \n",
      " Cube size : 268\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_29,2) with id: 34  \n",
      " Cube size : 268\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_645,2) with id: 35  \n",
      " Cube size : 267\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_113,2) with id: 36  \n",
      " Cube size : 266\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_533,2) with id: 37  \n",
      " Cube size : 266\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_743,2) with id: 38  \n",
      " Cube size : 265\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_720,2) with id: 39  \n",
      " Cube size : 264\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_108,2) with id: 40  \n",
      " Cube size : 263\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_66,2) with id: 41  \n",
      " Cube size : 263\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_44,2) with id: 42  \n",
      " Cube size : 262\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_754,2) with id: 43  \n",
      " Cube size : 261\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_647,2) with id: 44  \n",
      " Cube size : 260\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_80,2) with id: 45  \n",
      " Cube size : 260\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_62,2) with id: 46  \n",
      " Cube size : 259\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_2,2) with id: 47  \n",
      " Cube size : 259\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_419,2) with id: 48  \n",
      " Cube size : 259\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_112,2) with id: 49  \n",
      " Cube size : 259\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_7,2) with id: 50  \n",
      " Cube size : 259\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_672,2) with id: 51  \n",
      " Cube size : 259\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_695,2) with id: 52  \n",
      " Cube size : 258\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_390,2) with id: 53  \n",
      " Cube size : 257\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_164,2) with id: 54  \n",
      " Cube size : 257\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_224,2) with id: 55  \n",
      " Cube size : 257\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_310,2) with id: 56  \n",
      " Cube size : 257\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_31,2) with id: 57  \n",
      " Cube size : 256\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_717,2) with id: 58  \n",
      " Cube size : 256\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_418,2) with id: 59  \n",
      " Cube size : 255\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_779,2) with id: 60  \n",
      " Cube size : 254\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_82,2) with id: 61  \n",
      " Cube size : 253\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_25,2) with id: 62  \n",
      " Cube size : 252\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_91,2) with id: 63  \n",
      " Cube size : 252\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_499,2) with id: 64  \n",
      " Cube size : 252\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_32,2) with id: 65  \n",
      " Cube size : 252\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_531,2) with id: 66  \n",
      " Cube size : 251\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_764,2) with id: 67  \n",
      " Cube size : 250\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_223,2) with id: 68  \n",
      " Cube size : 249\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_55,2) with id: 69  \n",
      " Cube size : 249\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_225,2) with id: 70  \n",
      " Cube size : 249\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_362,2) with id: 71  \n",
      " Cube size : 249\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_364,2) with id: 72  \n",
      " Cube size : 249\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_282,2) with id: 73  \n",
      " Cube size : 248\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_79,2) with id: 74  \n",
      " Cube size : 248\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_763,2) with id: 75  \n",
      " Cube size : 248\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_27,2) with id: 76  \n",
      " Cube size : 247\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_506,2) with id: 77  \n",
      " Cube size : 247\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_10,2) with id: 78  \n",
      " Cube size : 247\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_711,2) with id: 79  \n",
      " Cube size : 247\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_675,2) with id: 80  \n",
      " Cube size : 246\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_760,2) with id: 81  \n",
      " Cube size : 246\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_168,2) with id: 82  \n",
      " Cube size : 246\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_615,2) with id: 83  \n",
      " Cube size : 246\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_141,2) with id: 84  \n",
      " Cube size : 246\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_778,2) with id: 85  \n",
      " Cube size : 246\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_780,2) with id: 86  \n",
      " Cube size : 245\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_53,2) with id: 87  \n",
      " Cube size : 245\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_41,2) with id: 88  \n",
      " Cube size : 245\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_532,2) with id: 89  \n",
      " Cube size : 245\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_705,2) with id: 90  \n",
      " Cube size : 245\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_23,2) with id: 91  \n",
      " Cube size : 244\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_195,2) with id: 92  \n",
      " Cube size : 243\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_765,2) with id: 93  \n",
      " Cube size : 243\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_220,2) with id: 94  \n",
      " Cube size : 242\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_43,2) with id: 95  \n",
      " Cube size : 242\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_74,2) with id: 96  \n",
      " Cube size : 241\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_741,2) with id: 97  \n",
      " Cube size : 241\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_92,2) with id: 98  \n",
      " Cube size : 240\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_505,2) with id: 99  \n",
      " Cube size : 240\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_21,2) with id: 100  \n",
      " Cube size : 239\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_277,2) with id: 101  \n",
      " Cube size : 239\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_716,2) with id: 102  \n",
      " Cube size : 239\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_735,2) with id: 103  \n",
      " Cube size : 238\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_106,2) with id: 104  \n",
      " Cube size : 237\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_700,2) with id: 105  \n",
      " Cube size : 237\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_694,2) with id: 106  \n",
      " Cube size : 237\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_89,2) with id: 107  \n",
      " Cube size : 236\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_35,2) with id: 108  \n",
      " Cube size : 236\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_47,2) with id: 109  \n",
      " Cube size : 236\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_88,2) with id: 110  \n",
      " Cube size : 236\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_674,2) with id: 111  \n",
      " Cube size : 236\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_746,2) with id: 112  \n",
      " Cube size : 236\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_137,2) with id: 113  \n",
      " Cube size : 235\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_78,2) with id: 114  \n",
      " Cube size : 235\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_365,2) with id: 115  \n",
      " Cube size : 235\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_722,2) with id: 116  \n",
      " Cube size : 235\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_52,2) with id: 117  \n",
      " Cube size : 234\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: u_354,2 with id: 118  \n",
      " Cube size : 234\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_475,2) with id: 119  \n",
      " Cube size : 234\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_773,2) with id: 120  \n",
      " Cube size : 234\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_737,2) with id: 121  \n",
      " Cube size : 233\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_1,2) with id: 122  \n",
      " Cube size : 232\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_361,2) with id: 123  \n",
      " Cube size : 231\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_221,2) with id: 124  \n",
      " Cube size : 231\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_337,2) with id: 125  \n",
      " Cube size : 231\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_0,2) with id: 126  \n",
      " Cube size : 231\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_504,2) with id: 127  \n",
      " Cube size : 231\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_49,2) with id: 128  \n",
      " Cube size : 231\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_59,2) with id: 129  \n",
      " Cube size : 230\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_502,2) with id: 130  \n",
      " Cube size : 230\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_84,2) with id: 131  \n",
      " Cube size : 229\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_718,2) with id: 132  \n",
      " Cube size : 228\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_4,2) with id: 133  \n",
      " Cube size : 227\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_393,2) with id: 134  \n",
      " Cube size : 227\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_26,2) with id: 135  \n",
      " Cube size : 227\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_279,2) with id: 136  \n",
      " Cube size : 227\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_739,2) with id: 137  \n",
      " Cube size : 227\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_756,2) with id: 138  \n",
      " Cube size : 226\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_165,2) with id: 139  \n",
      " Cube size : 225\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_14,2) with id: 140  \n",
      " Cube size : 225\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_56,2) with id: 141  \n",
      " Cube size : 225\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_219,2) with id: 142  \n",
      " Cube size : 225\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_328,2) with id: 143  \n",
      " Cube size : 225\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_753,2) with id: 144  \n",
      " Cube size : 224\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_719,2) with id: 145  \n",
      " Cube size : 224\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_63,2) with id: 146  \n",
      " Cube size : 223\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_125,2) with id: 147  \n",
      " Cube size : 223\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_733,2) with id: 148  \n",
      " Cube size : 222\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_51,2) with id: 149  \n",
      " Cube size : 221\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_477,2) with id: 150  \n",
      " Cube size : 220\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_734,2) with id: 151  \n",
      " Cube size : 219\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_278,2) with id: 152  \n",
      " Cube size : 218\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_171,2) with id: 153  \n",
      " Cube size : 218\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_762,2) with id: 154  \n",
      " Cube size : 217\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_37,2) with id: 155  \n",
      " Cube size : 216\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_736,2) with id: 156  \n",
      " Cube size : 215\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_646,2) with id: 157  \n",
      " Cube size : 214\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_728,2) with id: 158  \n",
      " Cube size : 214\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_222,2) with id: 159  \n",
      " Cube size : 214\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_33,2) with id: 160  \n",
      " Cube size : 214\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_281,2) with id: 161  \n",
      " Cube size : 213\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_559,2) with id: 162  \n",
      " Cube size : 212\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_713,2) with id: 163  \n",
      " Cube size : 211\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_476,2) with id: 164  \n",
      " Cube size : 210\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_424,2) with id: 165  \n",
      " Cube size : 210\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_446,2) with id: 166  \n",
      " Cube size : 209\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_449,2) with id: 167  \n",
      " Cube size : 208\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_305,2) with id: 168  \n",
      " Cube size : 207\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_254,2) with id: 169  \n",
      " Cube size : 207\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_698,2) with id: 170  \n",
      " Cube size : 206\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_30,2) with id: 171  \n",
      " Cube size : 205\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_6,2) with id: 172  \n",
      " Cube size : 205\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_75,2) with id: 173  \n",
      " Cube size : 205\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_701,2) with id: 174  \n",
      " Cube size : 205\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_249,2) with id: 175  \n",
      " Cube size : 205\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_775,2) with id: 176  \n",
      " Cube size : 205\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_116,2) with id: 177  \n",
      " Cube size : 204\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_57,2) with id: 178  \n",
      " Cube size : 204\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_60,2) with id: 179  \n",
      " Cube size : 204\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_46,2) with id: 180  \n",
      " Cube size : 203\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_68,2) with id: 181  \n",
      " Cube size : 203\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_40,2) with id: 182  \n",
      " Cube size : 202\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_564,2) with id: 183  \n",
      " Cube size : 202\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_136,2) with id: 184  \n",
      " Cube size : 202\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_194,2) with id: 185  \n",
      " Cube size : 202\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_70,2) with id: 186  \n",
      " Cube size : 202\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_149,2) with id: 187  \n",
      " Cube size : 201\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_283,2) with id: 188  \n",
      " Cube size : 201\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_77,2) with id: 189  \n",
      " Cube size : 200\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_589,2) with id: 190  \n",
      " Cube size : 200\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_774,2) with id: 191  \n",
      " Cube size : 200\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_104,2) with id: 192  \n",
      " Cube size : 199\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_39,2) with id: 193  \n",
      " Cube size : 199\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_166,2) with id: 194  \n",
      " Cube size : 198\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_253,2) with id: 195  \n",
      " Cube size : 198\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_58,2) with id: 196  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_587,2) with id: 197  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_169,2) with id: 198  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_782,2) with id: 199  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: u_489,2 with id: 200  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_17,2) with id: 201  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_781,2) with id: 202  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_85,2) with id: 203  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_19,2) with id: 204  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_250,2) with id: 205  \n",
      " Cube size : 197\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_338,2) with id: 206  \n",
      " Cube size : 197\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_170,2) with id: 207  \n",
      " Cube size : 196\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_148,2) with id: 208  \n",
      " Cube size : 196\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_90,2) with id: 209  \n",
      " Cube size : 196\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_726,2) with id: 210  \n",
      " Cube size : 196\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_755,2) with id: 211  \n",
      " Cube size : 196\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_640,2) with id: 212  \n",
      " Cube size : 195\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_12,2) with id: 213  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_730,2) with id: 214  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_727,2) with id: 215  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_561,2) with id: 216  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_392,2) with id: 217  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_761,2) with id: 218  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_333,2) with id: 219  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_193,2) with id: 220  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_109,2) with id: 221  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_20,2) with id: 222  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_420,2) with id: 223  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_671,2) with id: 224  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_560,2) with id: 225  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_334,2) with id: 226  \n",
      " Cube size : 194\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_67,2) with id: 227  \n",
      " Cube size : 194\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_48,2) with id: 228  \n",
      " Cube size : 193\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_226,2) with id: 229  \n",
      " Cube size : 193\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_36,2) with id: 230  \n",
      " Cube size : 192\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_776,2) with id: 231  \n",
      " Cube size : 192\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_81,2) with id: 232  \n",
      " Cube size : 191\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_563,2) with id: 233  \n",
      " Cube size : 191\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_309,2) with id: 234  \n",
      " Cube size : 191\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_86,2) with id: 235  \n",
      " Cube size : 190\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_251,2) with id: 236  \n",
      " Cube size : 189\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_50,2) with id: 237  \n",
      " Cube size : 189\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_391,2) with id: 238  \n",
      " Cube size : 189\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_758,2) with id: 239  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_447,2) with id: 240  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_16,2) with id: 241  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_673,2) with id: 242  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_192,2) with id: 243  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_18,2) with id: 244  \n",
      " Cube size : 188\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_732,2) with id: 245  \n",
      " Cube size : 188\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_751,2) with id: 246  \n",
      " Cube size : 187\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_740,2) with id: 247  \n",
      " Cube size : 186\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_107,2) with id: 248  \n",
      " Cube size : 185\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_421,2) with id: 249  \n",
      " Cube size : 185\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_336,2) with id: 250  \n",
      " Cube size : 185\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_139,2) with id: 251  \n",
      " Cube size : 185\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_363,2) with id: 252  \n",
      " Cube size : 184\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_24,2) with id: 253  \n",
      " Cube size : 183\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_742,2) with id: 254  \n",
      " Cube size : 183\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_770,2) with id: 255  \n",
      " Cube size : 182\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_703,2) with id: 256  \n",
      " Cube size : 181\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_725,2) with id: 257  \n",
      " Cube size : 181\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_738,2) with id: 258  \n",
      " Cube size : 180\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_8,2) with id: 259  \n",
      " Cube size : 179\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_752,2) with id: 260  \n",
      " Cube size : 179\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_280,2) with id: 261  \n",
      " Cube size : 178\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_69,2) with id: 262  \n",
      " Cube size : 177\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_643,2) with id: 263  \n",
      " Cube size : 176\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_45,2) with id: 264  \n",
      " Cube size : 176\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_783,2) with id: 265  \n",
      " Cube size : 176\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_450,2) with id: 266  \n",
      " Cube size : 175\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_198,2) with id: 267  \n",
      " Cube size : 174\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_721,2) with id: 268  \n",
      " Cube size : 174\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_503,2) with id: 269  \n",
      " Cube size : 173\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_771,2) with id: 270  \n",
      " Cube size : 173\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_444,2) with id: 271  \n",
      " Cube size : 172\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_772,2) with id: 272  \n",
      " Cube size : 172\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_142,2) with id: 273  \n",
      " Cube size : 171\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_110,2) with id: 274  \n",
      " Cube size : 170\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_617,2) with id: 275  \n",
      " Cube size : 170\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_276,2) with id: 276  \n",
      " Cube size : 169\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_448,2) with id: 277  \n",
      " Cube size : 169\n",
      " sat \n",
      " Literal was not removed \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "------------------------------------\n",
      " Searching Literal: Not(u_474,2) with id: 278  \n",
      " Cube size : 169\n",
      " unsat \n",
      " ******************* \n",
      " Literal was removed \n",
      " ******************* \n",
      "------------------------------------\n",
      "\n",
      "\n",
      "Total search time: 18.0851s\n"
     ]
    }
   ],
   "source": [
    "abdXP, sat_time = bestSNN_BCM.computeAXP(args[\"img_ind\"], args[\"time_slice\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "08bc0752",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " Abductive explanation found of length 169 in 18.09s time \n"
     ]
    }
   ],
   "source": [
    "print(\" Abductive explanation found of length {} in {:.2f}s time \".format(len(abdXP), sat_time))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "928795d5",
   "metadata": {},
   "source": [
    "### Visualise found Abductive explanation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "ecfe5083",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define two distinct arrays to store the active and non-active pixels in the abductive explanation\n",
    "\n",
    "def splitPixl(abdXP):\n",
    "    \n",
    "    # Split the abductive explanation pixels into active, non-active groups\n",
    "    \n",
    "    array = [] # active neurons\n",
    "    not_array = [] # non-active neurons\n",
    "\n",
    "    for literal in abdXP:\n",
    "        literal_name = repr(literal).split(\"(\")\n",
    "        if len(literal_name) == 1:\n",
    "            split_num = literal_name[0].split(\"_\")[1].split(\",\")[0]\n",
    "            array.append(int(split_num))\n",
    "\n",
    "        else:\n",
    "            split_num = literal_name[1].split(\"_\")[1].split(\",\")[0]\n",
    "            not_array.append(int(split_num))\n",
    "            \n",
    "            \n",
    "    return array, not_array\n",
    "\n",
    "\n",
    "array, not_array = splitPixl(abdXP)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fa175574",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.patches as patches\n",
    "\n",
    "def single_plot_axp_userInput(img_ind, abdXP):\n",
    "\n",
    "    # abdXP, sat_time = bestSNN_BCM.computeAXP(img_ind, time_slice, verbose=0)\n",
    "    array, not_array = splitPixl(abdXP)\n",
    "\n",
    "    array_color = 1\n",
    "    not_array_color = 0.3\n",
    "\n",
    "    image_pixels, _ = test_dataset[img_ind]\n",
    "    plt.figure(figsize=(7,7))\n",
    "\n",
    "    plt.subplot(1, 3, 1)\n",
    "    image_pixels = image_pixels.reshape((28, 28))\n",
    "    plt.title(\"Image shown\")\n",
    "    plt.imshow(image_pixels, cmap='gray')\n",
    "\n",
    "    plt.subplot(1, 3, 2)\n",
    "    abdXP_flattened = torch.as_tensor([array_color if i in array else not_array_color if i in not_array else 0 for i in range(784)])\n",
    "    abdXP_map = torch.reshape(abdXP_flattened, (28, 28))\n",
    "    plt.title(\"AXP found\")\n",
    "    plt.imshow(abdXP_map, cmap='gray')\n",
    "\n",
    "    # Third subplot: Highlight AXP pixels on the original image with red boxes\n",
    "    plt.subplot(1, 3, 3)\n",
    "    plt.title(\"AXP on the\\n digit image\")\n",
    "    plt.imshow(image_pixels, cmap='gray')\n",
    "\n",
    "    # Overlay red boxes for the highlighted pixels\n",
    "    ax = plt.gca()\n",
    "    \n",
    "    for i in range(28):\n",
    "        for j in range(28):\n",
    "            if abdXP_map[i, j] == array_color:  # Highlight if pixel is in AXP map\n",
    "                # Draw a rectangle around the pixel (i, j)\n",
    "                rect = patches.Rectangle((j - 0.5, i - 0.5), 1, 1, linewidth=1, edgecolor='red', facecolor='none')\n",
    "                ax.add_patch(rect)\n",
    "\n",
    "            elif abdXP_map[i, j] == not_array_color:  # Highlight if pixel is in AXP map\n",
    "                # Draw a rectangle around the pixel (i, j)\n",
    "                rect = patches.Rectangle((j - 0.5, i - 0.5), 1, 1, linewidth=1, edgecolor='yellow', facecolor='none')\n",
    "                ax.add_patch(rect)\n",
    "\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "print(\"Visualising found AXP for image index: {} at all times\".format(args['img_ind']))\n",
    "single_plot_axp_userInput(args['img_ind'], abdXP)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 318,
   "id": "1feec7c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.patches as patches\n",
    "\n",
    "def single_plot_axp(img_ind, time_slice):\n",
    "\n",
    "    abdXP, sat_time = bestSNN_BCM.computeAXP(img_ind, time_slice, verbose=0)\n",
    "    array, not_array = splitPixl(abdXP)\n",
    "\n",
    "    array_color = 1\n",
    "    not_array_color = 0.3\n",
    "\n",
    "    image_pixels, _ = test_dataset[img_ind]\n",
    "    out_spike = bestSNN_BCM.spike_dict_layer2[time_slice][img_ind]\n",
    "    plt.figure(figsize=(7,7))\n",
    "\n",
    "    plt.subplot(1, 3, 1)\n",
    "    image_pixels = image_pixels.reshape((28, 28))\n",
    "    image_timeslice = image_pixels*torch.reshape(torch.as_tensor(bestSNN_BCM.input_spike_dict[time_slice][img_ind]), (28,28))\n",
    "    plt.title(\"Image shown\\n at time step {}\".format(time_slice))\n",
    "    plt.imshow(image_timeslice, cmap='gray')\n",
    "\n",
    "    plt.subplot(1, 3, 2)\n",
    "    abdXP_flattened = torch.as_tensor([array_color if i in array else not_array_color if i in not_array else 0 for i in range(784)])\n",
    "    abdXP_map = torch.reshape(abdXP_flattened, (28, 28))\n",
    "    plt.title(\"AXP\\n at time step {}\".format(time_slice))\n",
    "    plt.imshow(abdXP_map, cmap='gray')\n",
    "\n",
    "    # Third subplot: Highlight AXP pixels on the original image with red boxes\n",
    "    plt.subplot(1, 3, 3)\n",
    "    plt.title(\"AXP match on digit\\n output\\n {} \\n at time step {}\".format(out_spike, time_slice))\n",
    "    plt.imshow(image_timeslice, cmap='gray')\n",
    "\n",
    "    # Overlay red boxes for the highlighted pixels\n",
    "    ax = plt.gca()\n",
    "    \n",
    "    for i in range(28):\n",
    "        for j in range(28):\n",
    "            if abdXP_map[i, j] == array_color:  # Highlight if pixel is in AXP map\n",
    "                # Draw a rectangle around the pixel (i, j)\n",
    "                rect = patches.Rectangle((j - 0.5, i - 0.5), 1, 1, linewidth=1, edgecolor='red', facecolor='none')\n",
    "                ax.add_patch(rect)\n",
    "\n",
    "            elif abdXP_map[i, j] == not_array_color:  # Highlight if pixel is in AXP map\n",
    "                # Draw a rectangle around the pixel (i, j)\n",
    "                rect = patches.Rectangle((j - 0.5, i - 0.5), 1, 1, linewidth=1, edgecolor='yellow', facecolor='none')\n",
    "                ax.add_patch(rect)\n",
    "\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d435b4e6",
   "metadata": {},
   "source": [
    "# Verify the found abductive explanation "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "id": "88647a20",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Function to verify if the computed explanation is a Prime Implicant\n",
    "\n",
    "def checkPI(abdXP, out_spike_slice, time_slice, equArr):\n",
    "    \n",
    "    abdXP_copy = abdXP[:]\n",
    "\n",
    "    flag_unsat = False\n",
    "    \n",
    "    for l_id, lit in enumerate(abdXP_copy):\n",
    "\n",
    "        explanandum = And(*[ Bool(f'v2_{i},{time_slice}') if bool(out_spike_slice[i]) else Not(Bool(f'v2_{i},{time_slice}')) for i in range(len(out_spike_slice)) ])\n",
    "        \n",
    "        abdXP_copy_2 = abdXP_copy[:]\n",
    "        \n",
    "        solver = Solver()\n",
    "\n",
    "        abdXP_copy_2.pop(l_id)\n",
    "        \n",
    "        # Add the initialisation constraint at time slice = -1\n",
    "        initConst = [Not(Bool(f'u_{i},{-1}')) for i in range(args['input-size']) ]\n",
    "        initConst.extend([Not(Bool(f'v1_{i},{-1}')) for i in range(args['hidden_neurons']) ])\n",
    "        initConst.extend([Not(Bool(f'v2_{i},{-1}')) for i in range(args['num_classes']) ])\n",
    "        \n",
    "        explanant = And(*abdXP_copy_2)\n",
    "\n",
    "        solver.add(And(*initConst))\n",
    "\n",
    "        beta1 = Or(Not(And(And(*equArr), explanant)), explanandum)\n",
    "\n",
    "        solver.add(Not(beta1))\n",
    "\n",
    "        if repr(solver.check()) == 'unsat':\n",
    "            flag_unsat = True\n",
    "            print(\"Not PI without lit {} at index {}\".format(lit, l_id))\n",
    "\n",
    "\n",
    "\n",
    "    if flag_unsat == True:\n",
    "        print(\"Abductive explanation is not a prime implicant\")\n",
    "\n",
    "    else:\n",
    "        print(\"It is an abductive explanation\")\n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "id": "86b6262f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "It is an abductive explanation\n"
     ]
    }
   ],
   "source": [
    "out_spike_slice = bestSNN_BCM.spike_dict_layer2[args[\"time_slice\"]][args[\"img_ind\"]]\n",
    "\n",
    "equArr = bestSNN_BCM.getEquSlice(args[\"time_slice\"], args[\"img_ind\"])\n",
    "\n",
    "checkPI(abdXP, out_spike_slice, args[\"time_slice\"], equArr)"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.15"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
