{"nbformat":4,"nbformat_minor":0,"metadata":{"colab":{"provenance":[],"gpuType":"T4","authorship_tag":"ABX9TyMAbnZgIZ4QcK8lyWAHyTje"},"kernelspec":{"name":"python3","display_name":"Python 3"},"language_info":{"name":"python"},"accelerator":"GPU"},"cells":[{"cell_type":"code","source":["!pip install pyeda"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"CMOWbjGrkuCY","executionInfo":{"status":"ok","timestamp":1747219532412,"user_tz":-540,"elapsed":20240,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"2c5504a5-79ef-49e5-c6a1-80cc3e4926aa"},"execution_count":1,"outputs":[{"output_type":"stream","name":"stdout","text":["Collecting pyeda\n","  Downloading pyeda-0.29.0.tar.gz (486 kB)\n","\u001b[?25l     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/486.8 kB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m486.8/486.8 kB\u001b[0m \u001b[31m31.0 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n","\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n","Building wheels for collected packages: pyeda\n","  Building wheel for pyeda (setup.py) ... \u001b[?25l\u001b[?25hdone\n","  Created wheel for pyeda: filename=pyeda-0.29.0-cp311-cp311-linux_x86_64.whl size=625897 sha256=4da0d17dc3b05237033ce46589829ceab074a28d636c18603580dc49e48de7fa\n","  Stored in directory: /root/.cache/pip/wheels/0e/95/33/49ca56535609f00570aa1da4ec98297ec06828361c0a37051f\n","Successfully built pyeda\n","Installing collected packages: pyeda\n","Successfully installed pyeda-0.29.0\n"]}]},{"cell_type":"code","execution_count":2,"metadata":{"id":"TWXy9_D0VAZt","executionInfo":{"status":"ok","timestamp":1747219545346,"user_tz":-540,"elapsed":12933,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"outputs":[],"source":["import torch\n","import torch.nn as nn\n","import torch.nn.functional as F\n","from torchvision import datasets, transforms\n","from torch.utils.data import DataLoader\n","\n","def get_mnist_splits(batch_size=64):\n","    tf = transforms.ToTensor()\n","    train = datasets.MNIST(root=\"./data\", train=True, download=True, transform=tf)\n","    test = datasets.MNIST(root=\"./data\", train=False, download=True, transform=tf)\n","\n","    val_size = 10000\n","    train_data, val_data = torch.utils.data.random_split(train, [len(train) - val_size, val_size])\n","\n","    return (\n","        DataLoader(train_data, batch_size=batch_size, shuffle=True),\n","        DataLoader(val_data, batch_size=batch_size),\n","        DataLoader(test, batch_size=batch_size)\n","    )\n"]},{"cell_type":"code","source":["class Operand_selector(nn.Module):\n","    def __init__(self, x, y):\n","        super().__init__()\n","        self.p = nn.Linear(x, y, bias=False)\n","\n","    def forward(self, x):\n","        w = self.p.weight\n","        mask = torch.zeros_like(w).scatter_(1, w.argmax(dim=-1, keepdim=True), 1.0)\n","        masked = mask - w.detach() + w\n","        return F.linear(x, masked)\n","\n","def bin_op(a, b, i):\n","    ops = [\n","        lambda a, b: torch.zeros_like(a),\n","        lambda a, b: a * b,\n","        lambda a, b: a - a * b,\n","        lambda a, b: a,\n","        lambda a, b: b - a * b,\n","        lambda a, b: b,\n","        lambda a, b: a + b - 2 * a * b,\n","        lambda a, b: a + b - a * b,\n","        lambda a, b: 1 - (a + b - a * b),\n","        lambda a, b: 1 - (a + b - 2 * a * b),\n","        lambda a, b: 1 - b,\n","        lambda a, b: 1 - b + a * b,\n","        lambda a, b: 1 - a,\n","        lambda a, b: 1 - a + a * b,\n","        lambda a, b: 1 - a * b,\n","        lambda a, b: torch.ones_like(a)\n","    ]\n","    return ops[i](a, b)\n","\n","class RoundSTE(torch.autograd.Function):\n","    @staticmethod\n","    def forward(ctx, input):\n","        return torch.round(input)\n","    @staticmethod\n","    def backward(ctx, grad_output):\n","        return grad_output\n","\n","def bin_op_s(a, b, i_s):\n","    r = torch.zeros_like(a)\n","    for i in range(16):\n","        r += i_s[..., i] * bin_op(a, b, i)\n","    return RoundSTE.apply(r)\n","\n","class Operator(nn.Module):\n","    def __init__(self, y):\n","        super().__init__()\n","        self.weights = nn.Parameter(torch.randn(y, 16))\n","\n","    def forward(self, a, b):\n","        w = self.weights\n","        mask = torch.zeros_like(w).scatter_(1, w.argmax(dim=-1, keepdim=True), 1.0)\n","        masked = mask - w.detach() + w\n","        return bin_op_s(a, b, masked)\n","\n","class oslgn(nn.Module):\n","    def __init__(self, x, y):\n","        super().__init__()\n","        self.os1 = Operand_selector(x, y)\n","        self.os2 = Operand_selector(x, y)\n","        self.op = Operator(y)\n","\n","    def forward(self, x):\n","        return self.op(self.os1(x), self.os2(x))\n","\n","class OSLGN_Depth(nn.Module):\n","    def __init__(self, input_dim, hidden_dim, depth):\n","        super().__init__()\n","        self.layers = nn.ModuleList()\n","        dims = [input_dim] + [hidden_dim] * (depth - 1) + [10]\n","        for i in range(depth):\n","            self.layers.append(oslgn(dims[i], dims[i+1]))\n","\n","    def forward(self, x):\n","        x = x.flatten(1)\n","        for layer in self.layers:\n","            x = layer(x)\n","        out = F.one_hot(x.argmax(dim=1), num_classes=10).float()\n","        return out - x.detach() + x\n"],"metadata":{"id":"Q70PSicCVjSY","executionInfo":{"status":"ok","timestamp":1747219545348,"user_tz":-540,"elapsed":1,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":3,"outputs":[]},{"cell_type":"code","source":["@torch.no_grad()\n","def evaluate(model, loader, criterion):\n","    model.eval()\n","    total_loss, correct, total = 0.0, 0, 0\n","    for x, y in loader:\n","        x, y = x.cuda(), y.cuda()\n","        out = model(x)\n","        loss = criterion(out, y)\n","        total_loss += loss.item() * x.size(0)\n","        correct += (out.argmax(dim=1) == y).sum().item()\n","        total += x.size(0)\n","    return correct / total, total_loss / total\n","\n","def train_epoch(model, loader, opt, criterion):\n","    model.train()\n","    total_loss = 0.0\n","    for x, y in loader:\n","        x, y = x.cuda(), y.cuda()\n","        opt.zero_grad()\n","        out = model(x)\n","        loss = criterion(out, y)\n","        loss.backward()\n","        opt.step()\n","        total_loss += loss.item() * x.size(0)\n","    return total_loss / len(loader.dataset)\n"],"metadata":{"id":"TGWPns_UWHLZ","executionInfo":{"status":"ok","timestamp":1747219545349,"user_tz":-540,"elapsed":0,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":4,"outputs":[]},{"cell_type":"code","source":["train_loader, val_loader, _ = get_mnist_splits()\n","model = OSLGN_Depth(784, 512, depth=4).cuda()\n","opt = torch.optim.Adam(model.parameters(), lr=1e-3)\n","criterion = nn.CrossEntropyLoss()\n","\n","for epoch in range(10):\n","    train_loss = train_epoch(model, train_loader, opt, criterion)\n","    val_acc, val_loss = evaluate(model, val_loader, criterion)\n","    print(f\"[Epoch {epoch+1}] Train Loss: {train_loss:.4f} | Val Acc: {val_acc:.4f} | Val Loss: {val_loss:.4f}\")\n","\n","torch.save(model.state_dict(), \"oslgn_depth4_epoch10.pth\")\n","print(\"Model saved to oslgn_depth4_epoch10.pth\")"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"VBJd5Z13WH6p","executionInfo":{"status":"ok","timestamp":1747219766775,"user_tz":-540,"elapsed":221425,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"b5e84591-e14e-4ac2-a808-87dd6dd96371"},"execution_count":5,"outputs":[{"output_type":"stream","name":"stderr","text":["100%|██████████| 9.91M/9.91M [00:01<00:00, 5.03MB/s]\n","100%|██████████| 28.9k/28.9k [00:00<00:00, 133kB/s]\n","100%|██████████| 1.65M/1.65M [00:01<00:00, 1.27MB/s]\n","100%|██████████| 4.54k/4.54k [00:00<00:00, 6.45MB/s]\n"]},{"output_type":"stream","name":"stdout","text":["[Epoch 1] Train Loss: 2.0832 | Val Acc: 0.4539 | Val Loss: 2.0073\n","[Epoch 2] Train Loss: 1.9846 | Val Acc: 0.5076 | Val Loss: 1.9536\n","[Epoch 3] Train Loss: 1.9869 | Val Acc: 0.4260 | Val Loss: 2.0352\n","[Epoch 4] Train Loss: 1.9770 | Val Acc: 0.4976 | Val Loss: 1.9636\n","[Epoch 5] Train Loss: 1.9639 | Val Acc: 0.5236 | Val Loss: 1.9376\n","[Epoch 6] Train Loss: 1.9629 | Val Acc: 0.4830 | Val Loss: 1.9782\n","[Epoch 7] Train Loss: 1.9789 | Val Acc: 0.5121 | Val Loss: 1.9491\n","[Epoch 8] Train Loss: 1.9758 | Val Acc: 0.5093 | Val Loss: 1.9519\n","[Epoch 9] Train Loss: 1.9683 | Val Acc: 0.4759 | Val Loss: 1.9853\n","[Epoch 10] Train Loss: 1.9533 | Val Acc: 0.5462 | Val Loss: 1.9150\n","Model saved to oslgn_depth4_epoch10.pth\n"]}]},{"cell_type":"code","source":["import torch\n","\n","def extract_symbolic_structure(model):\n","    \"\"\"\n","    Extracts argmax-based symbolic logic structure from a trained OSLGN_Depth model.\n","    Returns a list of dictionaries with operator index and operand indices per layer.\n","    \"\"\"\n","    symbolic_layers = []\n","    for i, layer in enumerate(model.layers):\n","        layer_info = {}\n","\n","        # Operand selectors (argmax across input dim)\n","        w1 = layer.os1.p.weight.detach().cpu()\n","        w2 = layer.os2.p.weight.detach().cpu()\n","        op1_indices = w1.argmax(dim=1).tolist()\n","        op2_indices = w2.argmax(dim=1).tolist()\n","\n","        # Operator selector (16 binary ops)\n","        op_weight = layer.op.weights.detach().cpu()\n","        op_indices = op_weight.argmax(dim=1).tolist()\n","\n","        layer_info[\"layer\"] = i\n","        layer_info[\"operand1_indices\"] = op1_indices\n","        layer_info[\"operand2_indices\"] = op2_indices\n","        layer_info[\"operator_indices\"] = op_indices\n","\n","        symbolic_layers.append(layer_info)\n","\n","    return symbolic_layers\n"],"metadata":{"id":"SGrievWSW7SW","executionInfo":{"status":"ok","timestamp":1747219766776,"user_tz":-540,"elapsed":5,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":6,"outputs":[]},{"cell_type":"code","source":["model = OSLGN_Depth(784, 512, depth=4)\n","model.load_state_dict(torch.load(\"oslgn_depth4_epoch10.pth\"))\n","model.eval()\n","\n","symbolic_data = extract_symbolic_structure(model)\n","\n","# 예시 출력\n","for layer in symbolic_data:\n","    print(f\"Layer {layer['layer']}\")\n","    print(f\"Operators: {layer['operator_indices']}\")\n","    print(f\"Operands A: {layer['operand1_indices']}\")\n","    print(f\"Operands B: {layer['operand2_indices']}\")\n"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"924cuQMKX7iC","executionInfo":{"status":"ok","timestamp":1747219766885,"user_tz":-540,"elapsed":112,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"7ab0e254-f051-4d13-bb6f-ea9f8756cb48"},"execution_count":7,"outputs":[{"output_type":"stream","name":"stdout","text":["Layer 0\n","Operators: [10, 0, 14, 5, 9, 13, 14, 2, 8, 1, 13, 5, 4, 6, 12, 5, 10, 10, 5, 2, 11, 3, 7, 2, 7, 8, 1, 10, 6, 7, 0, 4, 9, 0, 7, 15, 10, 6, 3, 12, 5, 15, 0, 10, 5, 4, 7, 8, 0, 6, 8, 4, 11, 9, 11, 14, 11, 12, 5, 6, 0, 12, 15, 5, 1, 6, 6, 5, 0, 2, 1, 13, 11, 2, 11, 2, 3, 11, 12, 15, 12, 0, 9, 12, 0, 2, 4, 7, 13, 5, 15, 8, 7, 14, 9, 6, 12, 11, 8, 12, 3, 10, 8, 7, 7, 4, 12, 5, 7, 9, 12, 12, 9, 9, 15, 12, 8, 7, 15, 11, 0, 11, 1, 7, 3, 11, 6, 5, 9, 6, 13, 3, 4, 0, 9, 4, 0, 14, 8, 9, 9, 7, 4, 3, 1, 9, 3, 10, 10, 0, 14, 6, 6, 5, 1, 15, 12, 0, 2, 13, 2, 15, 2, 3, 5, 0, 14, 4, 11, 10, 2, 11, 2, 2, 1, 2, 0, 7, 5, 15, 12, 7, 3, 8, 6, 2, 0, 4, 1, 10, 5, 0, 14, 13, 10, 10, 4, 8, 10, 3, 3, 4, 7, 7, 3, 5, 11, 5, 1, 13, 8, 7, 11, 14, 8, 6, 12, 10, 7, 3, 10, 5, 9, 4, 11, 6, 8, 5, 8, 6, 13, 8, 2, 10, 9, 4, 15, 10, 8, 5, 2, 4, 1, 15, 1, 2, 9, 14, 1, 15, 7, 11, 12, 1, 2, 3, 8, 15, 13, 7, 13, 10, 6, 0, 10, 8, 14, 15, 5, 7, 4, 0, 2, 12, 14, 4, 7, 3, 0, 10, 12, 9, 9, 4, 7, 8, 12, 10, 8, 3, 3, 15, 8, 1, 15, 7, 12, 9, 7, 15, 12, 15, 4, 8, 9, 3, 12, 4, 2, 14, 10, 8, 11, 11, 12, 14, 13, 11, 9, 11, 8, 0, 7, 13, 6, 10, 7, 15, 5, 14, 2, 8, 7, 11, 0, 10, 0, 14, 7, 14, 14, 1, 15, 2, 9, 10, 6, 4, 1, 3, 14, 13, 6, 4, 3, 5, 0, 10, 4, 9, 3, 1, 2, 12, 3, 1, 3, 10, 6, 4, 11, 4, 2, 0, 14, 11, 14, 3, 8, 5, 4, 7, 7, 15, 15, 15, 2, 1, 15, 2, 10, 14, 2, 1, 13, 6, 1, 9, 6, 8, 9, 14, 4, 10, 2, 3, 4, 7, 11, 12, 7, 14, 4, 15, 10, 7, 14, 8, 1, 8, 11, 10, 12, 5, 2, 8, 13, 15, 0, 9, 14, 8, 10, 12, 0, 15, 10, 13, 2, 0, 6, 13, 1, 9, 9, 14, 2, 11, 3, 2, 10, 2, 14, 14, 9, 11, 1, 7, 8, 7, 12, 4, 1, 7, 15, 3, 6, 13, 3, 6, 12, 0, 9, 2, 1, 1, 13, 6, 8, 8, 10, 9, 11, 14, 1, 15, 14, 14, 5, 14, 6, 6, 1, 14, 11, 6, 10, 12, 12, 11, 15, 13, 10, 13, 10, 10, 10, 15, 9, 11, 3, 2]\n","Operands A: [680, 737, 159, 170, 176, 739, 768, 249, 255, 781, 263, 200, 690, 97, 345, 678, 523, 741, 165, 761, 661, 590, 342, 479, 103, 379, 540, 233, 221, 716, 704, 509, 395, 113, 458, 302, 602, 525, 490, 342, 166, 191, 626, 752, 662, 373, 228, 95, 370, 211, 321, 384, 576, 177, 511, 524, 652, 27, 103, 527, 71, 69, 559, 538, 721, 99, 191, 701, 551, 388, 278, 102, 387, 473, 476, 191, 343, 572, 186, 412, 405, 564, 198, 563, 260, 293, 436, 509, 338, 199, 726, 29, 644, 350, 181, 444, 347, 692, 87, 88, 224, 83, 399, 604, 416, 498, 98, 547, 152, 356, 135, 98, 273, 357, 663, 20, 232, 401, 662, 343, 350, 570, 453, 76, 333, 92, 686, 723, 69, 51, 556, 248, 332, 566, 511, 484, 444, 576, 674, 607, 342, 455, 154, 246, 220, 736, 609, 615, 723, 40, 563, 186, 347, 270, 253, 273, 651, 253, 100, 753, 343, 80, 177, 231, 114, 162, 80, 343, 415, 716, 17, 161, 149, 253, 311, 416, 197, 62, 192, 383, 560, 652, 426, 377, 59, 274, 720, 176, 582, 518, 680, 533, 129, 493, 52, 134, 590, 147, 409, 375, 649, 450, 243, 512, 107, 461, 534, 354, 173, 670, 38, 567, 415, 636, 556, 515, 580, 190, 462, 370, 366, 486, 651, 221, 379, 585, 528, 602, 96, 572, 442, 124, 666, 284, 174, 567, 639, 347, 768, 543, 479, 223, 699, 673, 398, 134, 184, 611, 780, 693, 482, 612, 516, 270, 332, 277, 77, 346, 37, 593, 295, 518, 248, 400, 25, 355, 55, 525, 235, 651, 343, 530, 590, 767, 678, 401, 184, 546, 430, 339, 101, 625, 271, 455, 488, 97, 445, 553, 484, 148, 698, 581, 206, 157, 314, 32, 611, 415, 116, 396, 744, 716, 291, 741, 99, 613, 120, 380, 90, 734, 776, 740, 183, 163, 52, 99, 706, 509, 255, 211, 539, 258, 319, 635, 132, 137, 100, 363, 239, 172, 719, 568, 369, 396, 182, 683, 53, 142, 261, 650, 387, 415, 311, 768, 99, 315, 106, 66, 397, 498, 663, 193, 177, 610, 99, 443, 428, 27, 139, 220, 417, 732, 76, 466, 162, 463, 781, 221, 527, 272, 351, 372, 347, 483, 54, 578, 124, 191, 254, 30, 383, 588, 597, 129, 60, 230, 45, 655, 567, 714, 583, 478, 10, 681, 149, 415, 516, 624, 596, 70, 97, 386, 509, 0, 273, 303, 767, 98, 133, 417, 622, 158, 562, 159, 550, 350, 611, 506, 113, 221, 347, 32, 380, 234, 555, 535, 249, 441, 560, 635, 301, 577, 650, 364, 575, 510, 414, 162, 356, 625, 153, 778, 536, 446, 282, 737, 250, 494, 42, 374, 187, 104, 513, 516, 203, 275, 184, 290, 638, 577, 683, 551, 597, 202, 370, 355, 317, 41, 538, 463, 472, 20, 193, 740, 503, 228, 568, 715, 443, 585, 722, 290, 698, 455, 643, 158, 667, 220, 404, 189, 486, 489, 713, 43, 279, 275, 5, 737, 344, 752, 228, 525, 560, 97, 461, 214, 735, 23, 475, 95, 57, 219]\n","Operands B: [187, 113, 595, 497, 526, 763, 657, 275, 451, 695, 733, 719, 107, 101, 291, 96, 633, 499, 454, 556, 735, 71, 453, 346, 74, 206, 666, 163, 190, 718, 1, 592, 376, 220, 549, 652, 420, 248, 379, 115, 488, 164, 611, 84, 693, 189, 591, 353, 481, 68, 259, 523, 355, 274, 249, 497, 177, 567, 37, 691, 516, 519, 216, 552, 2, 207, 276, 253, 566, 488, 177, 625, 640, 239, 779, 551, 657, 585, 396, 39, 11, 371, 674, 638, 765, 652, 727, 511, 428, 712, 332, 223, 227, 592, 378, 415, 322, 520, 531, 442, 33, 736, 442, 101, 444, 104, 107, 100, 179, 354, 121, 647, 583, 285, 91, 482, 342, 429, 574, 250, 299, 476, 12, 103, 626, 304, 714, 304, 41, 766, 571, 140, 258, 568, 441, 757, 756, 410, 170, 651, 628, 454, 233, 0, 125, 712, 327, 150, 177, 342, 363, 192, 234, 107, 602, 468, 768, 576, 217, 549, 272, 136, 43, 376, 637, 677, 528, 511, 229, 714, 543, 164, 127, 548, 379, 136, 292, 169, 319, 348, 474, 355, 646, 404, 35, 243, 51, 189, 121, 699, 325, 533, 72, 747, 331, 650, 668, 175, 645, 727, 531, 541, 293, 304, 38, 554, 188, 294, 752, 210, 391, 127, 272, 605, 133, 122, 689, 706, 407, 402, 442, 716, 270, 575, 297, 662, 505, 233, 97, 274, 181, 245, 125, 366, 679, 711, 159, 437, 561, 103, 181, 523, 132, 106, 208, 455, 184, 291, 175, 375, 484, 537, 327, 100, 378, 644, 722, 60, 427, 104, 227, 713, 688, 542, 555, 494, 634, 162, 483, 658, 127, 556, 376, 600, 778, 92, 91, 446, 758, 498, 135, 265, 655, 564, 101, 47, 301, 767, 484, 503, 670, 525, 316, 538, 781, 250, 565, 162, 477, 136, 449, 579, 657, 203, 621, 355, 340, 610, 370, 327, 539, 741, 347, 357, 37, 521, 247, 98, 767, 518, 540, 324, 276, 678, 249, 372, 102, 329, 103, 62, 260, 595, 712, 126, 588, 69, 186, 197, 438, 46, 550, 96, 348, 242, 376, 92, 105, 363, 192, 717, 506, 313, 290, 201, 306, 466, 368, 160, 596, 609, 484, 0, 596, 626, 748, 682, 662, 362, 145, 104, 442, 541, 556, 164, 747, 363, 68, 351, 420, 219, 220, 406, 528, 118, 146, 73, 147, 173, 10, 203, 101, 588, 166, 694, 49, 325, 42, 171, 250, 747, 100, 526, 66, 388, 77, 593, 53, 103, 52, 740, 656, 541, 517, 409, 237, 322, 410, 605, 662, 106, 720, 608, 291, 162, 442, 66, 612, 274, 670, 203, 660, 540, 657, 500, 248, 201, 71, 139, 587, 664, 741, 622, 303, 94, 23, 110, 247, 345, 705, 670, 730, 426, 99, 377, 434, 497, 303, 636, 242, 412, 240, 377, 340, 202, 206, 401, 290, 42, 219, 464, 59, 309, 548, 126, 461, 526, 408, 708, 102, 557, 276, 370, 117, 159, 757, 570, 77, 185, 341, 551, 342, 331, 297, 396, 608, 249, 305, 635, 770, 304, 579, 551, 466, 186, 217, 439, 692, 105, 528, 645, 780, 329]\n","Layer 1\n","Operators: [13, 1, 6, 4, 9, 9, 1, 2, 6, 5, 11, 13, 15, 5, 1, 3, 9, 9, 14, 10, 5, 8, 6, 0, 3, 1, 3, 9, 1, 8, 11, 6, 12, 4, 2, 14, 8, 7, 9, 0, 3, 8, 12, 8, 11, 1, 7, 11, 7, 11, 3, 6, 2, 9, 13, 8, 11, 2, 8, 11, 6, 15, 3, 8, 1, 8, 7, 0, 2, 13, 10, 14, 3, 7, 5, 14, 6, 2, 8, 8, 3, 15, 8, 1, 9, 5, 2, 10, 15, 0, 13, 4, 5, 12, 0, 4, 14, 15, 0, 3, 9, 6, 11, 14, 5, 9, 0, 4, 0, 13, 12, 12, 9, 13, 9, 8, 2, 1, 12, 4, 8, 1, 2, 4, 9, 14, 2, 2, 6, 1, 14, 8, 14, 7, 2, 15, 14, 0, 12, 6, 5, 4, 4, 8, 1, 9, 2, 12, 7, 6, 5, 10, 7, 1, 13, 12, 5, 4, 0, 13, 0, 8, 11, 3, 13, 5, 13, 0, 8, 14, 6, 9, 12, 14, 15, 14, 10, 1, 9, 0, 3, 7, 7, 14, 9, 7, 11, 11, 9, 1, 10, 15, 0, 7, 10, 7, 15, 9, 0, 5, 6, 5, 13, 1, 3, 1, 2, 7, 14, 14, 2, 8, 0, 5, 14, 7, 8, 8, 1, 10, 14, 5, 3, 1, 8, 4, 4, 7, 14, 3, 12, 2, 12, 10, 2, 11, 9, 7, 1, 0, 6, 2, 8, 2, 8, 7, 11, 8, 7, 13, 6, 6, 14, 5, 9, 1, 11, 12, 12, 6, 3, 1, 10, 3, 11, 3, 0, 1, 10, 5, 11, 0, 2, 8, 12, 4, 8, 11, 8, 15, 1, 0, 10, 8, 1, 7, 10, 11, 5, 11, 8, 1, 11, 8, 11, 13, 7, 14, 11, 4, 11, 13, 4, 2, 14, 10, 9, 10, 8, 1, 4, 10, 12, 7, 0, 0, 0, 8, 15, 2, 9, 2, 10, 15, 1, 3, 4, 5, 13, 14, 1, 4, 0, 10, 9, 5, 7, 3, 13, 11, 4, 14, 8, 8, 15, 11, 4, 5, 5, 6, 8, 9, 2, 7, 4, 1, 7, 4, 6, 1, 4, 3, 8, 6, 13, 2, 9, 10, 8, 0, 1, 0, 14, 2, 0, 10, 1, 5, 5, 9, 5, 11, 6, 15, 10, 0, 0, 9, 13, 11, 3, 5, 5, 14, 5, 11, 15, 1, 0, 2, 2, 8, 3, 9, 7, 12, 7, 6, 3, 14, 5, 1, 5, 4, 11, 7, 4, 10, 1, 10, 0, 8, 8, 14, 15, 10, 15, 1, 11, 1, 3, 14, 10, 12, 15, 5, 8, 11, 0, 5, 12, 6, 11, 12, 0, 12, 3, 7, 15, 0, 4, 3, 10, 5, 5, 13, 8, 0, 0, 7, 15, 2, 10, 12, 9, 4, 10, 8, 12, 10, 10, 4, 10, 13, 14, 12, 8, 11, 15, 9, 13, 11, 2, 12, 14, 0, 2, 8, 12, 3, 3, 15, 5, 10, 6, 5, 6, 5, 3, 3, 5, 9, 12, 11, 15, 3, 10, 4, 1, 13, 2, 10]\n","Operands A: [257, 306, 202, 21, 305, 107, 364, 280, 354, 162, 155, 45, 11, 266, 393, 508, 463, 79, 178, 450, 218, 201, 354, 216, 316, 400, 436, 358, 389, 33, 481, 28, 1, 364, 13, 421, 374, 113, 221, 56, 136, 255, 324, 371, 418, 92, 75, 49, 121, 92, 110, 338, 132, 296, 67, 339, 389, 88, 123, 133, 451, 150, 389, 13, 107, 281, 350, 303, 371, 245, 183, 488, 152, 350, 380, 363, 107, 309, 328, 284, 158, 184, 491, 473, 381, 474, 395, 117, 130, 307, 467, 178, 153, 7, 157, 276, 334, 226, 479, 510, 102, 97, 54, 344, 272, 149, 509, 477, 249, 354, 109, 466, 436, 87, 245, 7, 95, 282, 177, 440, 187, 477, 482, 200, 69, 38, 315, 368, 365, 412, 49, 86, 429, 213, 70, 368, 505, 20, 266, 109, 397, 89, 90, 334, 382, 211, 265, 63, 454, 172, 187, 407, 382, 225, 159, 449, 125, 138, 262, 108, 165, 201, 95, 391, 224, 182, 372, 26, 445, 298, 257, 15, 162, 255, 139, 40, 22, 164, 75, 256, 182, 508, 302, 209, 385, 73, 91, 245, 470, 468, 387, 279, 478, 352, 427, 378, 17, 13, 208, 208, 395, 360, 93, 121, 389, 415, 371, 422, 137, 146, 173, 511, 22, 386, 158, 129, 446, 434, 370, 458, 174, 251, 312, 1, 455, 328, 465, 199, 146, 164, 405, 218, 379, 284, 152, 55, 267, 416, 105, 254, 162, 23, 165, 113, 441, 59, 451, 238, 200, 134, 449, 355, 26, 443, 405, 470, 223, 233, 255, 172, 399, 108, 323, 154, 372, 463, 6, 117, 502, 192, 497, 345, 226, 358, 272, 453, 49, 486, 100, 207, 122, 113, 455, 135, 434, 200, 204, 169, 104, 164, 86, 265, 252, 405, 444, 41, 456, 308, 424, 218, 11, 247, 139, 348, 112, 242, 393, 347, 193, 67, 253, 343, 438, 99, 194, 496, 460, 228, 378, 62, 473, 275, 509, 342, 45, 289, 378, 324, 463, 130, 439, 364, 132, 30, 35, 41, 197, 289, 465, 78, 395, 427, 478, 131, 151, 371, 501, 274, 47, 126, 307, 131, 355, 306, 183, 322, 43, 21, 131, 468, 487, 362, 337, 473, 469, 123, 187, 265, 207, 401, 442, 461, 33, 229, 44, 300, 258, 425, 411, 370, 410, 375, 252, 279, 55, 236, 136, 442, 329, 215, 2, 111, 398, 322, 232, 409, 315, 143, 438, 141, 310, 169, 269, 358, 222, 423, 259, 323, 232, 183, 270, 331, 87, 35, 429, 111, 97, 170, 77, 168, 503, 329, 279, 22, 89, 460, 285, 15, 476, 488, 235, 148, 185, 235, 63, 117, 278, 13, 506, 59, 456, 310, 441, 49, 460, 405, 30, 372, 73, 256, 104, 206, 448, 63, 364, 208, 510, 68, 254, 421, 476, 305, 287, 155, 484, 497, 128, 336, 511, 153, 422, 282, 48, 116, 253, 150, 262, 165, 355, 356, 372, 17, 440, 182, 65, 0, 309, 489, 245, 304, 183, 502, 360, 21, 439, 252, 381, 464, 84, 154, 367, 182, 124, 207, 193, 418, 401, 385, 483, 398, 421, 276]\n","Operands B: [295, 445, 3, 426, 58, 369, 232, 287, 459, 493, 457, 182, 20, 410, 358, 152, 270, 325, 150, 322, 31, 412, 161, 45, 168, 106, 135, 241, 46, 333, 312, 22, 25, 505, 221, 175, 199, 49, 471, 169, 25, 511, 428, 146, 450, 71, 299, 78, 212, 393, 332, 0, 148, 483, 121, 276, 420, 218, 218, 241, 369, 496, 238, 239, 487, 317, 249, 34, 372, 324, 262, 170, 233, 365, 65, 490, 92, 98, 420, 309, 252, 116, 76, 475, 277, 66, 396, 340, 493, 420, 330, 51, 293, 172, 427, 15, 262, 382, 451, 13, 269, 95, 114, 232, 497, 308, 383, 138, 332, 245, 148, 471, 395, 282, 511, 355, 44, 222, 298, 121, 270, 164, 411, 21, 288, 420, 89, 168, 18, 412, 454, 269, 112, 385, 370, 510, 59, 14, 94, 211, 358, 408, 352, 81, 154, 106, 338, 153, 61, 211, 337, 335, 132, 386, 310, 127, 73, 456, 266, 10, 245, 68, 144, 87, 399, 305, 251, 379, 160, 414, 440, 15, 415, 128, 8, 214, 253, 377, 141, 83, 86, 477, 280, 389, 9, 244, 416, 369, 321, 353, 330, 421, 457, 152, 151, 323, 153, 172, 273, 416, 252, 41, 195, 20, 287, 108, 407, 171, 348, 410, 134, 423, 407, 267, 495, 400, 95, 108, 30, 362, 482, 57, 152, 428, 359, 442, 66, 37, 7, 170, 156, 192, 485, 448, 88, 66, 496, 169, 19, 323, 25, 35, 79, 196, 212, 218, 371, 248, 163, 103, 468, 108, 376, 38, 382, 251, 438, 393, 356, 333, 492, 308, 433, 22, 476, 205, 226, 399, 451, 165, 123, 402, 241, 84, 346, 291, 211, 437, 67, 299, 465, 254, 29, 295, 382, 283, 494, 351, 328, 19, 135, 460, 404, 123, 305, 164, 317, 66, 153, 250, 30, 401, 226, 141, 125, 13, 203, 153, 146, 22, 158, 196, 89, 486, 43, 354, 215, 412, 53, 477, 297, 200, 7, 328, 463, 80, 432, 60, 8, 509, 121, 185, 439, 437, 75, 167, 8, 352, 398, 413, 373, 457, 489, 490, 338, 89, 13, 421, 130, 503, 259, 44, 253, 462, 133, 322, 83, 125, 116, 468, 77, 133, 286, 365, 437, 324, 92, 359, 207, 54, 164, 149, 490, 410, 19, 206, 303, 442, 217, 15, 143, 204, 320, 284, 389, 284, 501, 305, 366, 216, 437, 242, 82, 457, 220, 426, 183, 438, 226, 172, 41, 101, 489, 107, 79, 29, 185, 412, 394, 211, 37, 18, 160, 224, 29, 156, 55, 182, 325, 451, 352, 459, 158, 492, 107, 13, 4, 9, 492, 373, 139, 107, 369, 115, 354, 28, 81, 353, 221, 495, 243, 252, 170, 18, 254, 185, 220, 13, 135, 403, 288, 415, 3, 142, 131, 289, 398, 488, 167, 478, 180, 425, 218, 101, 208, 248, 110, 406, 232, 172, 186, 422, 157, 70, 282, 435, 293, 435, 377, 488, 466, 343, 88, 223, 424, 62, 355, 445, 345, 358, 142, 93, 117, 470, 256, 503, 220, 107, 256, 298, 371, 249, 485, 274, 257, 114, 186, 63, 458, 374, 373, 295]\n","Layer 2\n","Operators: [7, 4, 10, 13, 3, 4, 11, 14, 2, 5, 9, 14, 10, 15, 10, 9, 1, 13, 5, 10, 15, 3, 4, 14, 7, 7, 2, 14, 6, 14, 13, 12, 2, 4, 11, 7, 5, 15, 9, 8, 1, 12, 2, 12, 10, 6, 7, 7, 8, 15, 2, 1, 3, 15, 14, 6, 3, 0, 10, 13, 4, 5, 5, 6, 5, 5, 11, 2, 10, 2, 10, 3, 3, 11, 7, 7, 3, 5, 10, 6, 8, 3, 4, 8, 6, 9, 1, 11, 3, 6, 10, 5, 12, 14, 14, 0, 9, 7, 1, 9, 3, 7, 6, 3, 11, 13, 11, 1, 15, 9, 14, 3, 11, 4, 8, 14, 6, 10, 12, 2, 13, 8, 11, 3, 14, 8, 0, 11, 10, 5, 10, 14, 4, 9, 6, 10, 15, 1, 1, 13, 4, 7, 3, 15, 1, 9, 11, 9, 13, 9, 11, 2, 2, 0, 15, 9, 0, 0, 5, 5, 9, 3, 10, 10, 11, 1, 15, 12, 4, 1, 7, 0, 2, 6, 9, 7, 1, 7, 8, 2, 14, 15, 10, 12, 6, 15, 6, 14, 13, 11, 10, 12, 0, 4, 4, 1, 0, 14, 6, 7, 0, 8, 10, 4, 0, 1, 3, 10, 7, 15, 8, 11, 12, 6, 15, 7, 1, 5, 8, 11, 10, 5, 13, 11, 0, 13, 7, 13, 12, 4, 14, 1, 15, 2, 14, 9, 10, 12, 7, 1, 10, 0, 0, 3, 8, 2, 13, 1, 6, 4, 11, 13, 5, 5, 11, 13, 11, 9, 12, 15, 0, 2, 1, 9, 12, 6, 12, 2, 2, 12, 11, 12, 10, 10, 11, 8, 8, 14, 7, 4, 7, 0, 14, 11, 5, 10, 5, 5, 3, 13, 6, 5, 8, 0, 9, 14, 9, 7, 14, 0, 12, 6, 12, 10, 8, 5, 5, 15, 5, 1, 12, 11, 15, 10, 11, 5, 10, 11, 10, 13, 13, 1, 7, 14, 15, 1, 13, 14, 10, 1, 6, 11, 6, 4, 0, 15, 5, 6, 12, 1, 15, 5, 9, 15, 13, 6, 4, 9, 5, 5, 12, 9, 10, 9, 7, 5, 7, 1, 1, 8, 6, 15, 11, 0, 1, 15, 3, 11, 6, 3, 11, 11, 10, 0, 2, 0, 9, 6, 7, 1, 9, 10, 10, 11, 7, 2, 0, 11, 5, 12, 6, 8, 14, 15, 15, 8, 5, 7, 4, 1, 8, 4, 15, 2, 13, 12, 12, 14, 12, 14, 5, 13, 8, 2, 1, 13, 5, 9, 14, 4, 9, 2, 7, 4, 13, 15, 14, 2, 6, 1, 2, 13, 3, 4, 14, 5, 6, 4, 15, 3, 15, 10, 9, 9, 13, 2, 15, 2, 11, 2, 5, 11, 6, 14, 7, 6, 4, 10, 9, 13, 12, 15, 1, 8, 5, 0, 6, 15, 5, 9, 5, 0, 10, 11, 14, 2, 2, 0, 11, 8, 14, 15, 5, 12, 11, 7, 2, 2, 11, 14, 13, 14, 1, 11, 0, 10, 6, 15, 5, 13, 9, 11, 10, 1, 13, 13, 9, 1, 4, 1, 10, 0]\n","Operands A: [17, 92, 362, 253, 69, 217, 416, 101, 146, 342, 21, 168, 391, 152, 382, 79, 75, 95, 57, 9, 193, 350, 495, 409, 145, 295, 149, 452, 309, 235, 6, 252, 187, 442, 27, 92, 41, 398, 92, 282, 420, 82, 164, 167, 297, 77, 426, 44, 384, 317, 320, 17, 68, 261, 495, 9, 381, 214, 33, 145, 176, 236, 306, 240, 402, 448, 333, 269, 455, 348, 392, 15, 53, 45, 118, 202, 51, 458, 429, 272, 299, 211, 337, 442, 351, 279, 239, 406, 385, 242, 489, 300, 213, 497, 243, 372, 267, 218, 129, 300, 299, 299, 8, 21, 107, 424, 340, 19, 471, 504, 107, 490, 124, 431, 369, 191, 78, 110, 29, 464, 126, 25, 111, 31, 304, 207, 128, 383, 370, 263, 461, 251, 115, 433, 4, 331, 321, 287, 24, 63, 396, 276, 44, 157, 189, 12, 445, 17, 11, 72, 368, 429, 26, 309, 401, 339, 272, 433, 128, 80, 142, 495, 403, 463, 425, 306, 211, 423, 173, 14, 341, 319, 375, 180, 503, 172, 178, 93, 205, 232, 164, 402, 77, 221, 331, 423, 105, 324, 403, 264, 14, 190, 245, 172, 415, 220, 411, 339, 223, 68, 313, 270, 474, 314, 204, 393, 334, 68, 18, 19, 355, 81, 375, 27, 23, 233, 303, 494, 133, 393, 385, 74, 38, 398, 137, 349, 33, 415, 511, 464, 26, 201, 52, 398, 236, 490, 309, 424, 493, 366, 210, 39, 505, 314, 482, 23, 216, 349, 331, 100, 367, 415, 196, 398, 323, 4, 116, 234, 68, 278, 76, 456, 473, 383, 305, 295, 348, 436, 511, 505, 228, 368, 140, 510, 335, 91, 49, 203, 489, 397, 236, 379, 21, 367, 208, 108, 119, 297, 120, 43, 381, 7, 30, 216, 359, 397, 196, 314, 270, 3, 302, 80, 251, 189, 187, 149, 479, 375, 388, 42, 308, 291, 413, 193, 508, 65, 125, 12, 127, 40, 72, 470, 33, 8, 92, 280, 414, 224, 244, 259, 322, 272, 468, 10, 174, 72, 104, 144, 499, 39, 147, 480, 399, 397, 444, 294, 277, 161, 54, 341, 491, 180, 274, 322, 439, 346, 80, 183, 177, 348, 8, 255, 171, 404, 359, 187, 331, 24, 276, 162, 363, 404, 236, 506, 130, 13, 432, 212, 139, 273, 332, 255, 183, 408, 437, 480, 300, 297, 14, 128, 225, 326, 165, 371, 243, 421, 397, 470, 121, 405, 207, 458, 307, 50, 477, 220, 336, 276, 120, 456, 241, 95, 310, 377, 51, 270, 152, 215, 206, 149, 25, 176, 355, 95, 30, 108, 160, 87, 136, 117, 415, 439, 183, 197, 132, 398, 454, 170, 224, 85, 226, 157, 24, 214, 401, 125, 136, 159, 3, 11, 416, 140, 188, 320, 494, 81, 430, 24, 194, 120, 403, 121, 190, 112, 407, 118, 107, 131, 374, 376, 326, 191, 6, 461, 169, 331, 229, 212, 261, 508, 299, 342, 123, 359, 434, 176, 258, 315, 12, 398, 430, 110, 2, 129, 259, 138, 363, 439, 158, 136, 437, 400, 42, 123, 193, 386, 56, 365, 299, 335, 206, 190]\n","Operands B: [49, 378, 473, 322, 338, 52, 273, 226, 193, 506, 391, 41, 333, 300, 312, 471, 178, 414, 373, 437, 252, 177, 107, 358, 273, 145, 62, 134, 306, 457, 115, 212, 310, 252, 13, 46, 277, 203, 151, 332, 413, 287, 241, 393, 69, 330, 364, 228, 454, 319, 249, 29, 156, 411, 482, 55, 278, 20, 480, 9, 17, 212, 17, 202, 170, 444, 115, 91, 460, 193, 383, 360, 463, 257, 391, 175, 331, 359, 158, 202, 468, 362, 44, 409, 489, 362, 309, 288, 34, 308, 168, 402, 270, 193, 141, 245, 389, 396, 129, 18, 135, 365, 355, 405, 117, 427, 130, 335, 292, 185, 249, 410, 6, 450, 225, 365, 135, 245, 308, 211, 84, 328, 246, 71, 488, 171, 484, 254, 348, 438, 394, 366, 410, 393, 301, 501, 383, 268, 8, 464, 55, 366, 333, 380, 167, 362, 43, 418, 237, 146, 273, 67, 397, 133, 348, 379, 246, 327, 337, 23, 460, 20, 151, 363, 163, 178, 122, 26, 107, 210, 430, 137, 489, 71, 261, 393, 39, 30, 285, 280, 380, 170, 404, 252, 474, 511, 218, 379, 259, 122, 103, 99, 299, 237, 190, 301, 55, 227, 278, 394, 324, 58, 26, 455, 463, 89, 390, 307, 470, 423, 375, 344, 501, 147, 240, 5, 71, 183, 426, 273, 256, 299, 477, 138, 498, 341, 11, 366, 49, 227, 237, 174, 339, 24, 232, 246, 260, 362, 495, 475, 112, 62, 198, 428, 330, 381, 199, 224, 324, 163, 49, 459, 245, 167, 81, 361, 161, 234, 428, 307, 470, 42, 15, 311, 250, 383, 456, 72, 104, 43, 278, 273, 348, 225, 487, 292, 154, 468, 60, 226, 476, 128, 220, 261, 399, 86, 422, 329, 248, 248, 225, 247, 55, 453, 84, 217, 186, 393, 113, 102, 104, 229, 382, 145, 336, 270, 252, 123, 43, 480, 70, 406, 365, 437, 475, 231, 245, 390, 331, 95, 309, 204, 159, 14, 257, 473, 300, 303, 116, 430, 305, 188, 215, 114, 213, 107, 260, 447, 246, 22, 54, 225, 426, 148, 18, 471, 84, 483, 411, 507, 418, 207, 19, 299, 345, 225, 48, 63, 86, 78, 425, 417, 224, 240, 472, 253, 286, 371, 99, 222, 171, 90, 245, 254, 457, 364, 59, 320, 209, 7, 128, 162, 195, 401, 187, 98, 389, 135, 302, 331, 105, 188, 210, 104, 256, 161, 239, 351, 230, 314, 14, 148, 183, 6, 221, 462, 238, 267, 215, 367, 262, 122, 302, 323, 99, 485, 315, 138, 89, 404, 391, 73, 34, 187, 139, 101, 474, 310, 415, 41, 49, 272, 351, 274, 207, 20, 156, 443, 49, 19, 122, 253, 134, 91, 388, 427, 207, 326, 67, 491, 452, 114, 348, 177, 48, 477, 238, 439, 116, 261, 19, 222, 439, 162, 231, 131, 370, 249, 357, 174, 199, 344, 71, 104, 200, 398, 476, 378, 471, 31, 297, 284, 199, 259, 422, 258, 498, 204, 206, 203, 499, 163, 379, 284, 201, 249, 2, 247, 115, 380, 47, 368, 113, 79, 162, 159, 367, 96, 284, 352, 88, 166]\n","Layer 3\n","Operators: [6, 5, 7, 12, 13, 4, 14, 3, 15, 13]\n","Operands A: [221, 303, 52, 178, 407, 98, 91, 111, 382, 26]\n","Operands B: [245, 8, 107, 272, 348, 279, 91, 236, 455, 329]\n"]}]},{"cell_type":"code","source":["# Thanks to https://github.com/Felix-Petersen/difflogic/blob/main/difflogic/functional.py\n","PETERSEN_LOGIC_EXPR = {\n","    0:  lambda a, b: \"(False)\",\n","    1:  lambda a, b: f\"({a} and {b})\",\n","    2:  lambda a, b: f\"({a} and not {b})\",         # not(A ⇒ B)\n","    3:  lambda a, b: f\"({a})\",\n","    4:  lambda a, b: f\"(not {a} and {b})\",         # not(B ⇒ A)\n","    5:  lambda a, b: f\"({b})\",\n","    6:  lambda a, b: f\"(({a} and not {b}) or ({not {a} and b}))\",               # A xor B\n","    7:  lambda a, b: f\"({a} or {b})\",\n","    8:  lambda a, b: f\"(not ({a} or {b}))\",\n","    9:  lambda a, b: f\"(not (({a} and not {b}) or ({not {a} and b})))\",\n","    10: lambda a, b: f\"(not {b})\",\n","    11: lambda a, b: f\"({a} or not {b})\",          # B ⇒ A\n","    12: lambda a, b: f\"(not {a})\",\n","    13: lambda a, b: f\"(not {a} or {b})\",          # A ⇒ B\n","    14: lambda a, b: f\"(not ({a} and {b}))\",\n","    15: lambda a, b: \"(True)\"\n","}\n","\n","def recursive_draw(idx, depth = len(symbolic_data)):\n","    if depth == 0:\n","        return f\"x[{idx}]\"\n","    cur = symbolic_data[depth - 1]['operator_indices'][idx]\n","    left_idx = symbolic_data[depth - 1]['operand1_indices'][idx]\n","    right_idx = symbolic_data[depth - 1]['operand2_indices'][idx]\n","\n","    left = recursive_draw(left_idx, depth - 1)\n","    right = recursive_draw(right_idx, depth - 1)\n","    return PETERSEN_LOGIC_EXPR[cur](left, right)"],"metadata":{"id":"YYtxkWIOaB4H","executionInfo":{"status":"ok","timestamp":1747220871431,"user_tz":-540,"elapsed":7,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":61,"outputs":[]},{"cell_type":"code","source":["print(recursive_draw(1))"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"pQsvH91lafId","executionInfo":{"status":"ok","timestamp":1747220873319,"user_tz":-540,"elapsed":3,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"3fcc3dee-ca1e-46db-ea4d-6fa19e20e3e4"},"execution_count":62,"outputs":[{"output_type":"stream","name":"stdout","text":["((((not (x[355] or x[494])) and not (x[261] or x[438])) and not (((x[177] and not x[290]) or (False)) or ((x[347] and not x[234]) or (False)))))\n"]}]},{"cell_type":"code","source":["logic_expressions = {}\n","for cls in range(10):\n","    expr = recursive_draw(cls)\n","    logic_expressions[cls] = expr\n","\n","for cls in range(10):\n","    print(f\"class_{cls}(x) = {logic_expressions[cls]}\")"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"AeLIJuHhqLKL","executionInfo":{"status":"ok","timestamp":1747220904646,"user_tz":-540,"elapsed":5,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"999a3324-2ff9-43f1-a931-c84b526edd32"},"execution_count":64,"outputs":[{"output_type":"stream","name":"stdout","text":["class_0(x) = ((((not (x[462] or x[407]) and (x[482] or x[484]))) and not ((False) and not ((x[578] or not x[363]) or not (x[107])))) or (False))\n","class_1(x) = ((((not (x[355] or x[494])) and not (x[261] or x[438])) and not (((x[177] and not x[290]) or (False)) or ((x[347] and not x[234]) or (False)))))\n","class_2(x) = ((((not x[372] and x[541]) and not (x[347] and not x[556]))) or ((not (x[319] or x[276])) and ((not x[343] and x[511]))))\n","class_3(x) = (not (not (((x[350] or x[322]) and (x[152] or x[179])) or ((x[649]) or (not x[455] and x[564])))))\n","class_4(x) = (not (not ((not (((x[211] and not x[68]) or (False)) or (x[567] or x[127]))) and ((x[401] or x[429]) and (not (x[70] or x[747]))))) or (((not (x[568] or x[595])) and (x[454]))))\n","class_5(x) = (not (((not x[562] and x[517]) and (not x[562] and x[517])) and ((not x[562] and x[517]) and (not x[562] and x[517]))) and (not ((x[246]) and (x[356] and not x[587])) and (not (x[355]) and ((x[191] and not x[276]) or (False)))))\n","class_6(x) = (not ((((x[651] or x[658]))) and (((x[651] or x[658])))))\n","class_7(x) = ((((not (x[377] or x[404])))))\n","class_8(x) = (True)\n","class_9(x) = (not ((((x[149] and not x[127]) and not (x[567] or x[127])) or (False)) and not ((x[714] and not x[203]))) or ((((x[149] and not x[127]) and not (x[396] or not x[126])) or (False)) and ((not x[567] and x[711]))))\n"]}]},{"cell_type":"code","source":["import re\n","\n","def convert_to_pyeda_compatible(expr_str):\n","    expr_str = expr_str.replace(\"not \", \"~\")\n","    expr_str = expr_str.replace(\" and \", \" & \")\n","    expr_str = expr_str.replace(\" or \", \" | \")\n","    expr_str = expr_str.replace(\" ^ \", \" ^ \")  # XOR는 유지\n","    expr_str = expr_str.replace(\"(True)\", \"1\")\n","    expr_str = expr_str.replace(\"(False)\", \"0\")\n","    expr_str = re.sub(r\"x\\[(\\d+)\\]\", r\"x_\\1\", expr_str)\n","    return expr_str\n"],"metadata":{"id":"HGiT3Yy-sTuD","executionInfo":{"status":"ok","timestamp":1747220913749,"user_tz":-540,"elapsed":2,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":65,"outputs":[]},{"cell_type":"code","source":["import re\n","from pyeda.inter import expr\n","\n","def recover_python_expr_from_pyeda(expr_str):\n","    \"\"\"\n","    Converts PyEDA expression string into Python-style infix logic string.\n","    Handles Or(...), And(...), ~x_123, and constants like 1/0.\n","    \"\"\"\n","\n","    def is_complement(obj):\n","        return hasattr(obj, 'x') and str(obj).startswith('~')\n","\n","    def is_operator(obj, op_name):\n","        return str(type(obj)).endswith(f\"{op_name}Op'>\")\n","\n","    def to_infix(e):\n","        # Constants\n","        if str(e) == \"1\":\n","            return \"True\"\n","        elif str(e) == \"0\":\n","            return \"False\"\n","        # NOT (~x)\n","        elif is_complement(e):\n","            return f\"(not {to_infix(e.x)})\"\n","        # AND\n","        elif is_operator(e, \"And\"):\n","            return \"(\" + \" and \".join(to_infix(arg) for arg in e.xs) + \")\"\n","        # OR\n","        elif is_operator(e, \"Or\"):\n","            return \"(\" + \" or \".join(to_infix(arg) for arg in e.xs) + \")\"\n","        # Variable\n","        else:\n","            return str(e)\n","\n","    def restore_indexed_vars(s):\n","        return re.sub(r\"x_(\\d+)\", r\"x[\\1]\", s)\n","\n","    try:\n","        parsed = expr(expr_str)\n","        infix = to_infix(parsed)\n","        infix = restore_indexed_vars(infix)\n","        return infix.replace(\"~\", \"not \")\n","    except Exception as ex:\n","        return f\"Failed to parse or convert: {ex}\"\n"],"metadata":{"id":"adcH6H05xLfM","executionInfo":{"status":"ok","timestamp":1747220915093,"user_tz":-540,"elapsed":14,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":66,"outputs":[]},{"cell_type":"code","source":["import re\n","\n","def count_tokens(expr):\n","    return len(re.findall(r'\\bnot\\b|\\band\\b|\\bor\\b', expr))\n","\n","\n","def compute_compression_ratio(original, compressed):\n","    tok_o = count_tokens(original)\n","    tok_c = count_tokens(compressed)\n","    ratio = tok_o / tok_c if tok_c > 0 else float('inf')\n","    percent = (1 - tok_c / tok_o) * 100 if tok_o > 0 else 0\n","    return {\n","        \"original_tokens\": tok_o,\n","        \"compressed_tokens\": tok_c,\n","        \"compression_ratio\": round(ratio, 2),\n","        \"compression_rate_percent\": round(percent, 2)\n","    }\n"],"metadata":{"id":"3PpUThjKvwkL","executionInfo":{"status":"ok","timestamp":1747221106181,"user_tz":-540,"elapsed":3,"user":{"displayName":"Nyang","userId":"06114252970420503354"}}},"execution_count":69,"outputs":[]},{"cell_type":"code","source":["from pyeda.inter import expr, espresso_exprs\n","\n","compressed_results = {}\n","\n","for cls in range(10):\n","    raw_expr = recursive_draw(cls)\n","    pyeda_input = convert_to_pyeda_compatible(raw_expr)\n","    try:\n","        e = expr(pyeda_input).to_dnf()\n","        simplified, = espresso_exprs(e)\n","        compressed = str(simplified)\n","        converted_back = recover_python_expr_from_pyeda(compressed)\n","    except Exception as ex:\n","        compressed = f\"Compression failed: {ex}\"\n","    compressed_results[cls] = {\n","        \"original\": raw_expr,\n","        \"converted\": pyeda_input,\n","        \"compressed\": converted_back\n","    }\n","\n","for cls in range(10):\n","    print(f\"\\n[Class {cls}]\")\n","    print(f\"Original   : {compressed_results[cls]['original']}\")\n","    # print(f\"converted   : {compressed_results[cls]['converted']}\")\n","    print(f\"Compressed : {compressed_results[cls]['compressed']}\")\n","    result = compute_compression_ratio(compressed_results[cls]['original'], compressed_results[cls]['compressed'])\n","    print(result)\n"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"sWgZ7eJZriud","executionInfo":{"status":"ok","timestamp":1747220992617,"user_tz":-540,"elapsed":8,"user":{"displayName":"Nyang","userId":"06114252970420503354"}},"outputId":"a76e13e4-af79-4f7c-c11d-e2d63775219a"},"execution_count":68,"outputs":[{"output_type":"stream","name":"stdout","text":["\n","[Class 0]\n","Original   : ((((not (x[462] or x[407]) and (x[482] or x[484]))) and not ((False) and not ((x[578] or not x[363]) or not (x[107])))) or (False))\n","Compressed : ((not x[462] and not x[407] and x[482]) or (not x[462] and not x[407] and x[484]))\n","{'original_tokens': 13, 'compressed_tokens': 9, 'compression_ratio': 1.44, 'compression_rate_percent': 30.77}\n","\n","[Class 1]\n","Original   : ((((not (x[355] or x[494])) and not (x[261] or x[438])) and not (((x[177] and not x[290]) or (False)) or ((x[347] and not x[234]) or (False)))))\n","Compressed : ((not x[355] and not x[494] and not x[261] and not x[438] and x[290] and x[234]) or (not x[355] and not x[494] and not x[261] and not x[438] and not x[177] and x[234]) or (not x[355] and not x[494] and not x[261] and not x[438] and not x[177] and not x[347]) or (not x[355] and not x[494] and not x[261] and not x[438] and x[290] and not x[347]))\n","{'original_tokens': 14, 'compressed_tokens': 43, 'compression_ratio': 0.33, 'compression_rate_percent': -207.14}\n","\n","[Class 2]\n","Original   : ((((not x[372] and x[541]) and not (x[347] and not x[556]))) or ((not (x[319] or x[276])) and ((not x[343] and x[511]))))\n","Compressed : ((not x[372] and x[541] and x[556]) or (not x[319] and not x[276] and not x[343] and x[511]) or (not x[347] and not x[372] and x[541]))\n","{'original_tokens': 12, 'compressed_tokens': 15, 'compression_ratio': 0.8, 'compression_rate_percent': -25.0}\n","\n","[Class 3]\n","Original   : (not (not (((x[350] or x[322]) and (x[152] or x[179])) or ((x[649]) or (not x[455] and x[564])))))\n","Compressed : (x[649] or (x[322] and x[152]) or (x[350] and x[152]) or (not x[455] and x[564]) or (x[350] and x[179]) or (x[322] and x[179]))\n","{'original_tokens': 9, 'compressed_tokens': 11, 'compression_ratio': 0.82, 'compression_rate_percent': -22.22}\n","\n","[Class 4]\n","Original   : (not (not ((not (((x[211] and not x[68]) or (False)) or (x[567] or x[127]))) and ((x[401] or x[429]) and (not (x[70] or x[747]))))) or (((not (x[568] or x[595])) and (x[454]))))\n","Compressed : ((x[68] and not x[567] and not x[127] and x[429] and not x[70] and not x[747]) or (not x[568] and not x[595] and x[454]) or (x[68] and not x[567] and not x[127] and x[401] and not x[70] and not x[747]) or (not x[211] and not x[567] and not x[127] and x[401] and not x[70] and not x[747]) or (not x[211] and not x[567] and not x[127] and x[429] and not x[70] and not x[747]))\n","{'original_tokens': 17, 'compressed_tokens': 46, 'compression_ratio': 0.37, 'compression_rate_percent': -170.59}\n","\n","[Class 5]\n","Original   : (not (((not x[562] and x[517]) and (not x[562] and x[517])) and ((not x[562] and x[517]) and (not x[562] and x[517]))) and (not ((x[246]) and (x[356] and not x[587])) and (not (x[355]) and ((x[191] and not x[276]) or (False)))))\n","Compressed : ((not x[355] and not x[276] and not x[517] and x[587] and x[191]) or (not x[355] and not x[276] and x[562] and not x[356] and x[191]) or (not x[355] and not x[276] and x[562] and x[587] and x[191]) or (not x[355] and not x[276] and x[562] and not x[246] and x[191]) or (not x[355] and not x[276] and not x[517] and not x[356] and x[191]) or (not x[355] and not x[276] and not x[517] and not x[246] and x[191]))\n","{'original_tokens': 23, 'compressed_tokens': 48, 'compression_ratio': 0.48, 'compression_rate_percent': -108.7}\n","\n","[Class 6]\n","Original   : (not ((((x[651] or x[658]))) and (((x[651] or x[658])))))\n","Compressed : (not x[651] and not x[658])\n","{'original_tokens': 4, 'compressed_tokens': 3, 'compression_ratio': 1.33, 'compression_rate_percent': 25.0}\n","\n","[Class 7]\n","Original   : ((((not (x[377] or x[404])))))\n","Compressed : (not x[377] and not x[404])\n","{'original_tokens': 2, 'compressed_tokens': 3, 'compression_ratio': 0.67, 'compression_rate_percent': -50.0}\n","\n","[Class 8]\n","Original   : (True)\n","Compressed : (not x[377] and not x[404])\n","{'original_tokens': 0, 'compressed_tokens': 3, 'compression_ratio': 0.0, 'compression_rate_percent': 0}\n","\n","[Class 9]\n","Original   : (not ((((x[149] and not x[127]) and not (x[567] or x[127])) or (False)) and not ((x[714] and not x[203]))) or ((((x[149] and not x[127]) and not (x[396] or not x[126])) or (False)) and ((not x[567] and x[711]))))\n","Compressed : (x[567] or x[127] or not x[149] or (x[714] and not x[203]) or (not x[396] and x[126] and x[711]))\n","{'original_tokens': 22, 'compressed_tokens': 10, 'compression_ratio': 2.2, 'compression_rate_percent': 54.55}\n"]}]}]}