All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.electric.database.hierarchy.EDatabase Maven / Gradle / Ivy

There is a newer version: 9.02-e
Show newest version
/* -*- tab-width: 4 -*-
 *
 * Electric(tm) VLSI Design System
 *
 * File: EDatabase.java
 * Written by: Dmitry Nadezhin, Sun Microsystems.
 *
 * Copyright (c) 2003, Oracle and/or its affiliates. All rights reserved.
 *
 * Electric(tm) is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or
 * (at your option) any later version.
 *
 * Electric(tm) is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Electric(tm); see the file COPYING.  If not, write to
 * the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
 * Boston, Mass 02111-1307, USA.
 */
package com.sun.electric.database.hierarchy;

import com.sun.electric.database.CellBackup;
import com.sun.electric.database.CellRevision;
import com.sun.electric.database.CellTree;
import com.sun.electric.database.Environment;
import com.sun.electric.database.LibraryBackup;
import com.sun.electric.database.Snapshot;
import com.sun.electric.database.geometry.ERectangle;
import com.sun.electric.database.id.CellId;
import com.sun.electric.database.id.CellUsage;
import com.sun.electric.database.id.IdManager;
import com.sun.electric.database.id.LibId;
import com.sun.electric.database.id.TechId;
import com.sun.electric.database.network.NetworkManager;
import com.sun.electric.database.text.Setting;
import com.sun.electric.database.topology.NodeInst;
import com.sun.electric.technology.TechPool;
import com.sun.electric.technology.Technology;
import com.sun.electric.technology.technologies.Artwork;
import com.sun.electric.technology.technologies.Generic;
import com.sun.electric.technology.technologies.Schematics;
import com.sun.electric.tool.Job;
import com.sun.electric.tool.Tool;
import com.sun.electric.tool.user.ActivityLogger;
import com.sun.electric.util.ElapseTimer;
import com.sun.electric.util.TextUtils;
import com.sun.electric.util.collections.ImmutableArrayList;

import java.util.ArrayList;
import java.util.BitSet;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.prefs.BackingStoreException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Electric run-time database is a graph of ElectricObjects.
 */
public class EDatabase {

    private static final Logger logger = LoggerFactory.getLogger("com.sun.electric.database");
    private static EDatabase serverDatabase;
    private static EDatabase clientDatabase;
    private static boolean checkExamine;

    public static EDatabase serverDatabase() {
        return serverDatabase;
    }

    public static EDatabase clientDatabase() {
        return clientDatabase;
    }

    public static EDatabase currentDatabase() {
        return Job.getUserInterface().getDatabase();
    }

    public static void setServerDatabase(EDatabase database) {
        serverDatabase = database;
    }

    public static void setClientDatabase(EDatabase database) {
        clientDatabase = database;
    }

    public static void setCheckExamine() {
        checkExamine = true;
    }
    /** IdManager which keeps Ids of objects in this database.*/
    private final IdManager idManager;
    /** The optional name of this EDatabase */
    private final String name;
    /** Environment of this EDatabase */
    private Environment environment;
    /** list of linked technologies indexed by techId. */
    private TechPool techPool;
    /** list of linked libraries indexed by libId. */
    private final ArrayList linkedLibs = new ArrayList();
    /** map of libraries sorted by name */
    final TreeMap libraries = new TreeMap(TextUtils.STRING_NUMBER_ORDER);
    /** static list of all linked cells indexed by CellId. */
    final ArrayList linkedCells = new ArrayList();
    /** Last snapshot */
    private Snapshot snapshot;
    /** True if database matches snapshot. */
    private boolean snapshotFresh;
    /** Flag set when database invariants failed. */
    private boolean invariantsFailed;
    /** Network manager for this database. */
    private final NetworkManager networkManager;
    /** Thread which locked database for writing. */
    private volatile Thread writingThread;
    /** True if writing thread can changing. */
    private boolean canChanging;
    /** True if writing thread can undoing. */
    private boolean canUndoing;
    /** Tool which initiated changing. */
    private Tool changingTool;

    public EDatabase(Environment environment) {
        this(environment.techPool.idManager.getInitialSnapshot().with(null, environment, (CellTree[]) null, null));
    }

    /** Creates a new instance of EDatabase */
    public EDatabase(Snapshot snapshot) {
        this(snapshot, null);
    }

    public EDatabase(Snapshot snapshot, String name) {
        idManager = snapshot.idManager;
        this.name = name;
        this.snapshot = idManager.getInitialSnapshot();
        environment = this.snapshot.environment;
        techPool = environment.techPool;
        snapshotFresh = true;
        lock(true);
        canUndoing = true;
        undo(snapshot);
        canUndoing = false;
        unlock();
        networkManager = new NetworkManager();
    }

    public IdManager getIdManager() {
        return idManager;
    }

    public Snapshot getInitialSnapshot() {
        return idManager.getInitialSnapshot();
    }

    public NetworkManager getNetworkManager() {
        return networkManager;
    }

    public void setToolSettings(Setting.RootGroup toolSettings) {
        Environment newEnvironment = backup().environment.withToolSettings(toolSettings);
        setEnvironment(newEnvironment);
    }

    public void addTech(Technology tech) {
        Environment newEnvironment = backup().environment.addTech(tech);
        setEnvironment(newEnvironment);
    }

    public void implementSettingChanges(Setting.SettingChangeBatch changeBatch) {
        Environment oldEnvironment = backup().environment;
        Environment newEnvironment = environment.withSettingChanges(changeBatch);
        setEnvironment(newEnvironment);
    }

    private void setEnvironment(Environment newEnvironment) {
        if (this.environment == newEnvironment) {
            return;
        }
        resize(newEnvironment);
    }

    /** Returns TechPool of this database */
    public Environment getEnvironment() {
        return environment;
    }

    /** Returns TechPool of this database */
    public TechPool getTechPool() {
        return techPool;
    }

    public Collection getTechnologies() {
        return techPool.values();
    }

    /**
     * Get Technology by TechId
     * TechId must belong to same IdManager as TechPool
     * @param techId TechId to find
     * @return Technology b given TechId or null
     * @throws IllegalArgumentException of TechId is not from this IdManager
     */
    public Technology getTech(TechId techId) {
        return techPool.getTech(techId);
    }

    /** Return Artwork technology in this database */
    public Artwork getArtwork() {
        return techPool.getArtwork();
    }

    /** Return Generic technology in this database */
    public Generic getGeneric() {
        return techPool.getGeneric();
    }

    /** Return Schematic technology in this database */
    public Schematics getSchematics() {
        return techPool.getSchematics();
    }

    public Map getSettings() {
        return environment.getSettings();
    }

    public Library getLib(LibId libId) {
        return getLib(libId.libIndex);
    }

    void addLib(Library lib) {
        int libIndex = lib.getId().libIndex;
        while (libIndex >= linkedLibs.size()) {
            linkedLibs.add(null);
        }
        Library oldLib = linkedLibs.set(libIndex, lib);
        assert oldLib == null;
        libraries.put(lib.getName(), lib);
    }

    void removeLib(LibId libId) {
        Library oldLib = linkedLibs.set(libId.libIndex, null);
        while (!linkedLibs.isEmpty() && linkedLibs.get(linkedLibs.size() - 1) == null) {
            linkedLibs.remove(linkedLibs.size() - 1);
        }
        libraries.remove(oldLib.getName());
    }

    public Cell getCell(CellId cellId) {
        return getCell(cellId.cellIndex);
    }

    void addCell(Cell cell) {
        int cellIndex = cell.getCellIndex();
        while (cellIndex >= linkedCells.size()) {
            linkedCells.add(null);
        }
        Cell oldCell = linkedCells.set(cellIndex, cell);
        assert oldCell == null;
    }

    void removeCell(CellId cellId) {
        Cell oldCell = linkedCells.set(cellId.cellIndex, null);
        assert oldCell != null;
        while (!linkedCells.isEmpty() && linkedCells.get(linkedCells.size() - 1) == null) {
            linkedCells.remove(linkedCells.size() - 1);
        }
    }

//    Technology getTech(int techIndex) { return techIndex < linkedTechs.size() ? linkedTechs.get(techIndex) : null; }
    Library getLib(int libIndex) {
        return libIndex < linkedLibs.size() ? linkedLibs.get(libIndex) : null;
    }

    Cell getCell(int cellIndex) {
        return cellIndex < linkedCells.size() ? linkedCells.get(cellIndex) : null;
    }

    /**
     * Locks the database.
     * Lock may be either exclusive (for writing) or shared (for reading).
     * @param exclusive true if lock is for writing.
     */
    public void lock(boolean exclusive) {
        assert writingThread == null;
        if (exclusive) {
            writingThread = Thread.currentThread();
        }
        canChanging = canUndoing = false;
    }

    /**
     * Unlocks the database.
     */
    public void unlock() {
        writingThread = null;
    }

    /**
     * Method to check whether changing of database is allowed by current thread.
     * @throws IllegalStateException if changes are not allowed.
     */
    public void checkChanging() {
        if (Thread.currentThread() == writingThread && canChanging) {
            return;
        }
        IllegalStateException e = new IllegalStateException("Database changes are forbidden");
        logger.warn("EDatabase.checkChanging", e);
        throw e;
    }

    /**
     * Method to check whether changing of whole database is allowed.
     * @throws IllegalStateException if changes are not allowed.
     */
    public void checkUndoing() {
        if (Thread.currentThread() == writingThread && canUndoing) {
            return;
        }
        IllegalStateException e = new IllegalStateException("Database undo is forbidden");
        logger.warn("EDatabse.checkUndoing", e);
        throw e;
    }

    /**
     * Method to check whether examining of database is allowed.
     */
    public void checkExamine() {
        if (checkExamine) {
            if (Job.getUserInterface().getDatabase() == this) {
                return;
            }
        } else {
            if (writingThread == null || Thread.currentThread() == writingThread) {
                return;
            }
        }
        IllegalStateException e = new IllegalStateException("Cuncurrent database examine");
//        e.printStackTrace();
        logger.info("EDatabase.checkExamine", e.getMessage(), e);
//        throw e;
    }

    /**
     * Low-level method to begin changes in database.
     * @param changingTool tool which initiated
     */
    public void lowLevelBeginChanging(Tool changingTool) {
        if (Thread.currentThread() != writingThread) {
            checkChanging();
        }
        canChanging = true;
        this.changingTool = changingTool;
    }

    /**
     * Low-level method to permit changes in database.
     */
    public void lowLevelEndChanging() {
        if (Thread.currentThread() != writingThread) {
            checkChanging();
        }
        changingTool = null;
        canChanging = false;
    }

    /**
     * Low-level method to permit undos in database.
     */
    public void lowLevelSetCanUndoing(boolean b) {
        if (Thread.currentThread() != writingThread) {
            checkUndoing();
        }
        canUndoing = b;
    }

    /**
     * Get list of cells contained in other libraries
     * that refer to cells contained in this library
     * @param elib to search for
     * @return list of cells referring to elements in this library
     */
    public Set findReferenceInCell(Library elib) {
        TreeSet set = new TreeSet();

        for (Library l : libraries.values()) {
            // skip itself
            if (l == elib) {
                continue;
            }

            for (Cell cell : l.cells.values()) {
                cell.findReferenceInCell(elib, set);
            }
        }
        return set;
    }

    /**
     * Method to find a Library with the specified name.
     * @param libName the name of the Library.
     * Note that this is the Library name, and not the Library file.
     * @return the Library, or null if there is no known Library by that name.
     */
    public Library findLibrary(String libName) {
        if (libName == null) {
            return null;
        }
        return libraries.get(libName);
//		Library lib = libraries.get(libName);
//		if (lib != null) return lib;
//
//		for (Library l : libraries.values())
//		{
//			if (l.getName().equalsIgnoreCase(libName))
//				return l;
//		}
//		return null;
    }

    /**
     * Method to return an iterator over all libraries.
     * @return an iterator over all libraries.
     */
    public Iterator getLibraries() {
        synchronized (libraries) {
            ArrayList librariesCopy = new ArrayList(libraries.values());
            return librariesCopy.iterator();
        }
    }

    /**
     * Method to return the number of libraries.
     * @return the number of libraries.
     */
    public int getNumLibraries() {
        return libraries.size();
    }

    /**
     * Method to return an iterator over all visible libraries.
     * @return an iterator over all visible libraries.
     */
    public List getVisibleLibraries() {
        synchronized (libraries) {
            ArrayList visibleLibraries = new ArrayList();
            for (Library lib : libraries.values()) {
                if (!lib.isHidden()) {
                    visibleLibraries.add(lib);
                }
            }
            return visibleLibraries;
        }
    }

    void unfreshSnapshot() {
        checkChanging();
        snapshotFresh = false;
    }

    private synchronized void setSnapshot(Snapshot snapshot, boolean fresh) {
        this.snapshot = snapshot;
        environment = snapshot.environment;
        techPool = environment.techPool;
        this.snapshotFresh = fresh;
        environment.activate();
    }

    /**
     * Low-level method to atomically get fresh snapshot.
     * @return fresh snapshot of the database, or null if not fresh snapshot exists.
     */
    public synchronized Snapshot getFreshSnapshot() {
        return snapshotFresh ? snapshot : null;
    }

    /**
     * Create Snapshot from the current state of Electric database.
     * @return snapshot of the current state of Electric database.
     * @throws IllegalStateException if recalculation of Snapshot is required in thread which is not enabled to do it.
     */
    public Snapshot backup() {
        if (snapshotFresh) {
            return snapshot;
        }
        checkChanging();
        return doBackup();
    }

    private Snapshot doBackup() {
//        long startTime = System.currentTimeMillis();
        assert techPool == snapshot.techPool;
        CellTree[] cellTrees = new CellTree[linkedCells.size()];
        boolean cellsChanged = cellTrees.length != snapshot.cellTrees.size();
        for (int cellIndex = 0; cellIndex < cellTrees.length; cellIndex++) {
            Cell cell = getCell(cellIndex);
            if (cell != null) {
                cellTrees[cellIndex] = cell.tree();
            }
            cellsChanged = cellsChanged || cellTrees[cellIndex] != snapshot.getCellTree(cellIndex);
        }
        if (!cellsChanged) {
            cellTrees = null;
        }

        LibraryBackup[] libBackups = new LibraryBackup[linkedLibs.size()];
        boolean libsChanged = libBackups.length != snapshot.libBackups.size();
        for (int libIndex = 0; libIndex < libBackups.length; libIndex++) {
            Library lib = linkedLibs.get(libIndex);
            LibraryBackup libBackup = lib != null ? lib.backup() : null;
            libBackups[libIndex] = libBackup;
            libsChanged = libsChanged || snapshot.libBackups.get(libIndex) != libBackup;
        }
        if (!libsChanged) {
            libBackups = null;
        }

        setSnapshot(snapshot.with(changingTool, environment, cellTrees, libBackups), true);
        for (CellTree cellTree : snapshot.cellTrees) {
            if (cellTree == null) {
                continue;
            }
            Cell cell = getCell(cellTree.top.cellRevision.d.cellId);
            assert cell.tree() == cellTree;
        }
//        long endTime = System.currentTimeMillis();
//        if (Job.getDebug()) System.out.println("backup took: " + (endTime - startTime) + " msec");
        return snapshot;
    }

    /**
     * Force database to specified state.
     * This method can recover corrupted database.
     * @param snapshot snapshot to recover.
     */
    public void recover(Snapshot snapshot) {
        long startTime = System.currentTimeMillis();
        setSnapshot(snapshot, false);
        recoverLibraries();
        recycleCells();
        BitSet recovered = new BitSet();
        for (CellBackup newBackup : snapshot.cellBackups) {
            if (newBackup != null) {
                recoverRecursively(newBackup.cellRevision.d.cellId, recovered);
            }
        }
        for (Library lib : libraries.values()) {
            lib.collectCells();
        }
        recoverCellGroups();
        snapshotFresh = true;
        long endTime = System.currentTimeMillis();
        if (Job.getDebug()) {
            System.out.println("recover took: " + (endTime - startTime) + " msec");
            checkInvariants();
        }
    }

    private void recoverRecursively(CellId cellId, BitSet recovered) {
        int cellIndex = cellId.cellIndex;
        if (recovered.get(cellIndex)) {
            return;
        }
        CellTree newTree = snapshot.getCellTree(cellId);
        CellBackup newBackup = newTree.top;
        CellRevision newRevision = newBackup.cellRevision;
        for (int i = 0, numUsages = cellId.numUsagesIn(); i < numUsages; i++) {
            CellUsage u = cellId.getUsageIn(i);
            if (newRevision.getInstCount(u) <= 0) {
                continue;
            }
            recoverRecursively(u.protoId, recovered);
        }
        Cell cell = getCell(cellId);
        cell.recover(newTree);
        recovered.set(cellIndex);
    }

    /**
     * Force database to specified state.
     * This method assumes that database is in valid state.
     * @param snapshot snapshot to undo.
     */
    public void undo(Snapshot snapshot) {
//        long startTime = System.currentTimeMillis();
        Snapshot oldSnapshot = backup();
        if (oldSnapshot == snapshot) {
            return;
        }
        setSnapshot(snapshot, false);
        boolean cellGroupsChanged = snapshot.cellGroupsProbablyChanged(oldSnapshot);
        if (oldSnapshot.libBackups != snapshot.libBackups) {
            recoverLibraries();
            cellGroupsChanged = true;
        }
        recycleCells();

        BitSet cellNamesChangedInLibrary = new BitSet();
        ImmutableArrayList cellBackups = snapshot.cellBackups;
        if (oldSnapshot.cellBackups.size() == cellBackups.size()) {
            for (int cellIndex = 0; cellIndex < cellBackups.size(); cellIndex++) {
                CellBackup oldBackup = oldSnapshot.getCell(cellIndex);
                CellBackup newBackup = snapshot.getCell(cellIndex);
                if (oldBackup == newBackup) {
                    continue;
                }
                if (oldBackup == null) {
                    cellNamesChangedInLibrary.set(newBackup.cellRevision.d.getLibId().libIndex);
                    assert cellGroupsChanged;
                } else if (newBackup == null) {
                    cellNamesChangedInLibrary.set(oldBackup.cellRevision.d.getLibId().libIndex);
                    assert cellGroupsChanged;
//                } else {
//                    boolean moved = oldBackup.d.getLibId() != newBackup.d.getLibId();
//                    if (moved || oldBackup.d.cellName != newBackup.d.cellName) {
//                        cellNamesChangedInLibrary.set(newBackup.d.getLibId().libIndex);
//                        cellNamesChangedInLibrary.set(oldBackup.d.getLibId().libIndex);
//                    }
//                    if (moved)
//                        cellGroupsChanged = true;
                }
            }
        } else {
            cellGroupsChanged = true;
            if (snapshot.libBackups.size() > 0) // Bug in BitSet.set(int,int) on Sun JDK
            {
                cellNamesChangedInLibrary.set(0, snapshot.libBackups.size());
            }
        }

        BitSet updated = new BitSet();
        BitSet exportsModified = new BitSet();
        BitSet boundsModified = new BitSet();
        for (CellBackup newBackup : snapshot.cellBackups) {
            if (newBackup != null) {
                undoRecursively(oldSnapshot, newBackup.cellRevision.d.cellId, updated, exportsModified, boundsModified);
            }
        }
        if (!cellNamesChangedInLibrary.isEmpty()) {
            for (Library lib : libraries.values()) {
                if (cellNamesChangedInLibrary.get(lib.getId().libIndex)) {
                    lib.collectCells();
                }
            }
        }
        if (cellGroupsChanged) {
            recoverCellGroups();
        }
        snapshotFresh = true;
//        long endTime = System.currentTimeMillis();
//        if (Job.getDebug()) {
//            System.out.println("undo took: " + (endTime - startTime) + " msec");
//            checkFresh(snapshot);
//        }
    }

    private void undoRecursively(Snapshot oldSnapshot, CellId cellId, BitSet updated, BitSet exportsModified, BitSet boundsModified) {
        int cellIndex = cellId.cellIndex;
        if (updated.get(cellIndex)) {
            return;
        }
        CellTree newTree = snapshot.getCellTree(cellId);
        CellBackup newBackup = newTree.top;
        CellRevision newRevision = newBackup.cellRevision;
        assert cellId != null;
        boolean subCellsExportsModified = false;
        boolean subCellsBoundsModified = false;
        for (int i = 0, numUsages = cellId.numUsagesIn(); i < numUsages; i++) {
            CellUsage u = cellId.getUsageIn(i);
            if (newRevision.getInstCount(u) <= 0) {
                continue;
            }
            undoRecursively(oldSnapshot, u.protoId, updated, exportsModified, boundsModified);
            int subCellIndex = u.protoId.cellIndex;
            if (exportsModified.get(subCellIndex)) {
                subCellsExportsModified = true;
            }
            if (boundsModified.get(subCellIndex)) {
                subCellsBoundsModified = true;
            }
        }
        Cell cell = getCell(cellId);
        CellRevision oldRevision = oldSnapshot.getCellRevision(cellId);
        ERectangle oldBounds = oldSnapshot.getCellBounds(cellId);
        cell.undo(newTree,
                subCellsExportsModified ? exportsModified : null,
                subCellsBoundsModified ? boundsModified : null);
        updated.set(cellIndex);
        if (oldRevision == null || !newRevision.sameExports(oldRevision)) {
            exportsModified.set(cellIndex);
        }
        if (oldRevision == null || snapshot.getCellBounds(cellId) != oldBounds) {
            boundsModified.set(cellIndex);
        }
    }

    /**
     * Resize database after Technology change.
     * This method assumes that database is in valid state.
     * @param environment new Environment
     */
    public void resize(Environment environment) {
//        long startTime = System.currentTimeMillis();
        backup();
//        this.environment = environment;
//        this.techPool = environment.techPool;
//        environment.activate();
        lowLevelSetCanUndoing(true);
        undo(snapshot.with(changingTool, environment));
        lowLevelSetCanUndoing(false);
        assert snapshotFresh;
//        long endTime = System.currentTimeMillis();
        if (Job.getDebug()) {
//            System.out.println("resize took: " + (endTime - startTime) + " msec");
            checkFresh(snapshot);
        }
    }

    private void recoverLibraries() {
        while (linkedLibs.size() > snapshot.libBackups.size()) {
            Library lib = linkedLibs.remove(linkedLibs.size() - 1);
            if (lib != null) {
                lib.cells.clear();
            }
        }
        while (linkedLibs.size() < snapshot.libBackups.size()) {
            linkedLibs.add(null);
        }
        for (int libIndex = 0; libIndex < snapshot.libBackups.size(); libIndex++) {
            LibraryBackup libBackup = snapshot.libBackups.get(libIndex);
            Library lib = linkedLibs.get(libIndex);
            if (libBackup == null && lib != null) {
                linkedLibs.set(libIndex, null);
            } else if (libBackup != null && lib == null) {
                linkedLibs.set(libIndex, new Library(this, libBackup.d));
            }
            /*
            } else {
            
            Library lib = linkedLibs.get(libIndex);
            String libName = lib.getName();
            if (!oldBackup.d.libName.equals(libName)) {
            Cell curCell = lib.getCurCell();
            lib.prefs = allPrefs.node(libName);
            lib.prefs.put("LIB", libName);
            lib.curCellPref = null;
            lib.setCurCell(curCell);
            }
             */
        }
        libraries.clear();
        for (int libIndex = 0; libIndex < snapshot.libBackups.size(); libIndex++) {
            LibraryBackup libBackup = snapshot.libBackups.get(libIndex);
            if (libBackup == null) {
                continue;
            }
            Library lib = linkedLibs.get(libIndex);
            lib.recover(libBackup);
            libraries.put(lib.getName(), lib);
        }
        /* ???
        if (curLib == null || !curLib.isLinked()) {
        curLib = null;
        for(Library lib: libraries.values()) {
        if (lib.isHidden()) continue;
        curLib = lib;
        break;
        }
        }
         */
    }

    private void recycleCells() {
        ImmutableArrayList cellBackups = snapshot.cellBackups;
        while (linkedCells.size() > cellBackups.size()) {
            linkedCells.remove(linkedCells.size() - 1);
        }
        while (linkedCells.size() < cellBackups.size()) {
            linkedCells.add(null);
        }
        for (int cellIndex = 0; cellIndex < cellBackups.size(); cellIndex++) {
            CellBackup newBackup = cellBackups.get(cellIndex);
            Cell cell = linkedCells.get(cellIndex);
            if (newBackup == null) {
                if (cell != null) {
                    linkedCells.set(cellIndex, null);
                }
            } else if (cell == null) {
                linkedCells.set(cellIndex, new Cell(this, newBackup.cellRevision.d));
            }
        }
    }

    private void recoverCellGroups() {
        ArrayList> groups = new ArrayList>();
        for (int cellIndex = 0; cellIndex < snapshot.cellBackups.size(); cellIndex++) {
            CellBackup cellBackup = snapshot.cellBackups.get(cellIndex);
            if (cellBackup == null) {
                continue;
            }
            Cell cell = getCell(cellIndex);
            assert cell != null;
            CellId cellId = cell.getId();
            int cellGroupIndex = snapshot.getCellGroupIndex(cellId);
            if (cellGroupIndex == groups.size()) {
                groups.add(new TreeSet());
            }
            groups.get(cellGroupIndex).add(cell);
        }
        for (int i = 0; i < groups.size(); i++) {
            new Cell.CellGroup(groups.get(i));
        }
    }

    /**
     * Method to save isExpanded status of NodeInsts in this Library to Preferences.
     */
    public void saveExpandStatus() throws BackingStoreException {
        for (Iterator lit = getLibraries(); lit.hasNext();) {
            Library lib = lit.next();
            for (Iterator it = lib.getCells(); it.hasNext();) {
                Cell cell = it.next();
                cell.saveExpandStatus();
            }
        }
    }

    /**
     * Add specified NodeInst to a set of nodes.
     * @param nodes a data structure to accumulate nodes
     * @param ni NodeInst to add
     */
    public void addToNodes(Map nodes, NodeInst ni) {
        if (ni.getDatabase() != this || !ni.isLinked()) {
            throw new IllegalArgumentException();
        }
        CellId cellId = ni.getParent().getId();
        BitSet nodesInCell = nodes.get(cellId);
        if (nodesInCell == null) {
            nodesInCell = new BitSet();
            nodes.put(cellId, nodesInCell);
        }
        nodesInCell.set(ni.getD().nodeId);
    }

    public void expandNodes(Map nodesToExpand) {
        for (Map.Entry e : nodesToExpand.entrySet()) {
            Cell cell = getCell(e.getKey());
            cell.expand(e.getValue());
        }
    }

    /**
     * Method to check invariants in all Libraries.
     * @return true if invariants are valid
     */
    public boolean checkInvariants() {
        try {
            long startTime = System.currentTimeMillis();
            idManager.checkInvariants();
            backup();
            snapshot.check();
            check();
            if (Job.getDebug()) {
                long endTime = System.currentTimeMillis();
                ElapseTimer et = ElapseTimer.createInstanceByValues(startTime, endTime);
                System.out.println("**** Check Invariants took " + et);
            }
            return true;
        } catch (Throwable e) {
            if (!invariantsFailed) {
                System.out.println("Exception checking database invariants");
                e.printStackTrace();
                ActivityLogger.logException(e);
                invariantsFailed = true;
            }
        }
        return false;
    }

    /**
     * Method to check invariants in this EDatabase.
     * @exception AssertionError if invariants are not valid
     */
    private void check() {
        assert techPool == environment.techPool;
        if (snapshotFresh) {
            assert environment == snapshot.environment;
            assert techPool == snapshot.techPool;
            assert linkedLibs.size() == snapshot.libBackups.size();
            assert linkedCells.size() == snapshot.cellBackups.size();
        }

        for (int libIndex = 0; libIndex < linkedLibs.size(); libIndex++) {
            Library lib = linkedLibs.get(libIndex);
            if (lib == null) {
                if (snapshotFresh) {
                    assert snapshot.libBackups.get(libIndex) == null;
                }
                continue;
            }
            assert lib.getId() == getIdManager().getLibId(libIndex);
            assert libraries.get(lib.getName()) == lib;
            lib.check();
            if (snapshotFresh) {
                assert lib.backup == snapshot.libBackups.get(libIndex);
            }
        }

        for (int cellIndex = 0; cellIndex < linkedCells.size(); cellIndex++) {
            Cell cell = linkedCells.get(cellIndex);
            if (cell == null) {
                if (snapshotFresh) {
                    assert snapshot.cellBackups.get(cellIndex) == null;
                }
                continue;
            }
            CellId cellId = cell.getId();
            assert cellId == idManager.getCellId(cellIndex);
            Library lib = cell.getLibrary();
            assert lib.cells.get(cell.getCellName()) == cell;
            cell.check();
            if (snapshotFresh) {
                assert cell.cellBackupFresh;
                assert cell.backup == snapshot.cellBackups.get(cellIndex);
                assert cell.getBounds() == snapshot.getCellBounds(cellId);
//                cell.checkBoundsCorrect();
            }
        }

//        TreeSet libNames = new TreeSet(String.CASE_INSENSITIVE_ORDER);
//        for (Map.Entry e : libraries.entrySet()) {
//            String libName = e.getKey();
//            Library lib = e.getValue();
//            assert libName == lib.getName();
//            assert linkedLibs.get(lib.getId().libIndex) == lib;
//
//            assert !libNames.contains(libName) : "case insensitive " + libName;
//            libNames.add(libName);
//        }

        if (snapshotFresh) {
            HashMap groupNums = new HashMap();
            for (int i = 0; i < snapshot.cellBackups.size(); i++) {
                CellBackup cellBackup = snapshot.getCell(i);
                if (cellBackup == null) {
                    continue;
                }
                CellId cellId = cellBackup.cellRevision.d.cellId;
                Cell cell = getCell(cellId);
                Cell.CellGroup cellGroup = cell.getCellGroup();
                Integer gn = groupNums.get(cellGroup);
                if (gn == null) {
                    gn = Integer.valueOf(groupNums.size());
                    groupNums.put(cellGroup, gn);
                }
                int groupIndex = gn.intValue();
                assert snapshot.getCellGroupIndex(cellId) == groupIndex;
                Cell mainSchematics = cellGroup.getMainSchematics();
                assert snapshot.getMainSchematics(cellId) == (mainSchematics != null ? mainSchematics.getId() : null);
            }
        }
    }

    /**
     * Checks that Electric database has the expected state.
     * @param expectedSnapshot expected state.
     */
    public void checkFresh(Snapshot expectedSnapshot) {
        assert snapshotFresh && snapshot == expectedSnapshot;
        check();
    }

    @Override
    public String toString() {
        return name != null ? name : super.toString();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy