/* KLayout Layout Viewer Copyright (C) 2006-2017 Matthias Koefferlein This program 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 2 of the License, or (at your option) any later version. This program 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 this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "gsiDecl.h" #include "dbRecursiveShapeIterator.h" #include "dbRegion.h" #include "tlGlobPattern.h" namespace gsi { // --------------------------------------------------------------- // db::RecursiveShapeIterator binding static db::RecursiveShapeIterator *new_si1 (const db::Layout &layout, const db::Cell &cell, unsigned int layer) { return new db::RecursiveShapeIterator (layout, cell, layer); } static db::RecursiveShapeIterator *new_si2 (const db::Layout &layout, const db::Cell &cell, const std::vector &layers) { return new db::RecursiveShapeIterator (layout, cell, layers); } static db::RecursiveShapeIterator *new_si3 (const db::Layout &layout, const db::Cell &cell, unsigned int layer, const db::Box &box, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layer, box, overlapping); } static db::RecursiveShapeIterator *new_si3a (const db::Layout &layout, const db::Cell &cell, unsigned int layer, const db::Box &box, const db::Region &excl_region, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layer, box, excl_region, overlapping); } static db::RecursiveShapeIterator *new_si3b (const db::Layout &layout, const db::Cell &cell, unsigned int layer, const db::Region ®ion, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layer, region, overlapping); } static db::RecursiveShapeIterator *new_si4 (const db::Layout &layout, const db::Cell &cell, const std::vector &layers, const db::Box &box, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layers, box, overlapping); } static db::RecursiveShapeIterator *new_si4a (const db::Layout &layout, const db::Cell &cell, const std::vector &layers, const db::Box &box, const db::Region &excl_region, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layers, box, excl_region, overlapping); } static db::RecursiveShapeIterator *new_si4b (const db::Layout &layout, const db::Cell &cell, const std::vector &layers, const db::Region ®ion, bool overlapping) { return new db::RecursiveShapeIterator (layout, cell, layers, region, overlapping); } static void select_cells1 (db::RecursiveShapeIterator *r, const std::vector &cells) { std::set cc; cc.insert (cells.begin (), cells.end ()); r->select_cells (cc); } static void select_cells2 (db::RecursiveShapeIterator *r, const std::string &pattern) { tl::GlobPattern p (pattern); std::set cc; for (db::Layout::const_iterator ci = r->layout ()->begin (); ci != r->layout ()->end (); ++ci) { if (p.match (r->layout ()->cell_name (ci->cell_index ()))) { cc.insert (ci->cell_index ()); } } r->select_cells (cc); } static void unselect_cells1 (db::RecursiveShapeIterator *r, const std::vector &cells) { std::set cc; cc.insert (cells.begin (), cells.end ()); r->unselect_cells (cc); } static void unselect_cells2 (db::RecursiveShapeIterator *r, const std::string &pattern) { tl::GlobPattern p (pattern); std::set cc; for (db::Layout::const_iterator ci = r->layout ()->begin (); ci != r->layout ()->end (); ++ci) { if (p.match (r->layout ()->cell_name (ci->cell_index ()))) { cc.insert (ci->cell_index ()); } } r->unselect_cells (cc); } Class decl_RecursiveShapeIterator ("RecursiveShapeIterator", gsi::constructor ("new", &new_si1, "@brief Creates a recursive, single-layer shape iterator.\n" "@args layout, cell, layer\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layer The layer (index) from which the shapes are taken\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layer given by the layer index in the \"layer\" parameter.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::constructor ("new", &new_si2, "@brief Creates a recursive, multi-layer shape iterator.\n" "@args layout, cell, layer\n" "@args layout, cell, layers\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layers The layer indexes from which the shapes are taken\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layers given by the layer indexes in the \"layers\" parameter.\n" "While iterating use the \\layer method to retrieve the layer of the current shape.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::constructor ("new", &new_si3, "@brief Creates a recursive, single-layer shape iterator with a region.\n" "@args layout, cell, layer, box, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layer The layer (index) from which the shapes are taken\n" "@param box The search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layer given by the layer index in the \"layer\" parameter.\n" "\n" "The search is confined to the region given by the \"box\" parameter. If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::constructor ("new", &new_si3a, "@brief Creates a recursive, single-layer shape iterator with a region.\n" "@args layout, cell, layer, box, excl_region, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layer The layer (index) from which the shapes are taken\n" "@param box The basic search region\n" "@param excl_region The region that is excluded from the basic search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layer given by the layer index in the \"layer\" parameter.\n" "\n" "The search is confined to the region given by the \"box\" and the \"excl_region\" parameter. " "The box specifies the basic region. From that basic region the parts specified with \"excl_region\" " "are excluded. The excluded region needs to be a rectilinear region.\n" "\n" "If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.25.\n" ) + gsi::constructor ("new", &new_si3b, "@brief Creates a recursive, single-layer shape iterator with a region.\n" "@args layout, cell, layer, region, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layer The layer (index) from which the shapes are taken\n" "@param region The search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layer given by the layer index in the \"layer\" parameter.\n" "\n" "The search is confined to the region given by the \"region\" parameter. The region needs to be a rectilinear region.\n" "If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.25.\n" ) + gsi::constructor ("new", &new_si4, "@brief Creates a recursive, multi-layer shape iterator with a region.\n" "@args layout, cell, layers, box, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layers The layer indexes from which the shapes are taken\n" "@param box The search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layers given by the layer indexes in the \"layers\" parameter.\n" "While iterating use the \\layer method to retrieve the layer of the current shape.\n" "\n" "The search is confined to the region given by the \"box\" parameter. If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::constructor ("new", &new_si4a, "@brief Creates a recursive, multi-layer shape iterator with a region.\n" "@args layout, cell, layers, box, excl_region, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layers The layer indexes from which the shapes are taken\n" "@param box The basic search region\n" "@param excl_region The region that is excluded from the basic search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layers given by the layer indexes in the \"layers\" parameter.\n" "While iterating use the \\layer method to retrieve the layer of the current shape.\n" "\n" "The search is confined to the region given by the \"box\" and the \"excl_region\" parameter. " "The box specifies the basic region. From that basic region the parts specified with \"excl_region\" " "are excluded. The excluded region needs to be a rectilinear region.\n" "\n" "If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::constructor ("new", &new_si4b, "@brief Creates a recursive, multi-layer shape iterator with a region.\n" "@args layout, cell, layers, region, overlapping\n" "@param layout The layout which shall be iterated\n" "@param cell The initial cell which shall be iterated (including it's children)\n" "@param layers The layer indexes from which the shapes are taken\n" "@param region The search region\n" "@param overlapping If set to true, shapes overlapping the search region are reported, otherwise touching is sufficient\n" "\n" "This constructor creates a new recursive shape iterator which delivers the shapes of " "the given cell plus it's children from the layers given by the layer indexes in the \"layers\" parameter.\n" "While iterating use the \\layer method to retrieve the layer of the current shape.\n" "\n" "The search is confined to the region given by the \"region\" parameter. The region needs to be a rectilinear region.\n" "If \"overlapping\" is true, shapes whose " "bounding box is overlapping the search region are reported. If \"overlapping\" is false, shapes whose " "bounding box is touching the search region are reported.\n" "\n" "This constructor has been introduced in version 0.23.\n" ) + gsi::method ("max_depth=", (void (db::RecursiveShapeIterator::*) (int)) &db::RecursiveShapeIterator::max_depth, "@brief Specify the maximum hierarchy depth to look into\n" "@args depth\n" "\n" "A depth of 0 instructs the iterator to deliver only shapes from the initial cell.\n" "The depth must be specified before the shapes are being retrieved.\n" "Setting the depth resets the iterator.\n" ) + gsi::method ("max_depth", (int (db::RecursiveShapeIterator::*) () const) &db::RecursiveShapeIterator::max_depth, "@brief Gets the maximum hierarchy depth\n" "\n" "See \\max_depth= for a description of that attribute.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("reset", &db::RecursiveShapeIterator::reset, "@brief Resets the iterator to the initial state\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("reset_selection", &db::RecursiveShapeIterator::reset_selection, "@brief Resets the selection to the default state\n" "\n" "In the initial state, the top cell and it's children are selected. Child cells can be switched on and off " "together with their sub-hierarchy using \\select_cells and \\unselect_cells.\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("layout", &db::RecursiveShapeIterator::layout, "@brief Gets the layout this iterator is connected to\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("top_cell", &db::RecursiveShapeIterator::top_cell, "@brief Gets the top cell this iterator is connected to\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("region", &db::RecursiveShapeIterator::region, "@brief Gets the region that is iterator is using\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("region=", &db::RecursiveShapeIterator::set_region, "@brief Sets the region that is iterator is using\n" "@args region\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("overlapping?", &db::RecursiveShapeIterator::overlapping, "@brief Gets a flag indicating whether overlapping shapes are selected when a region is used\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("overlapping=", &db::RecursiveShapeIterator::set_overlapping, "@brief Sets a flag indicating whether overlapping shapes are selected when a region is used\n" "@args region\n" "\n" "If this flag is false, shapes touching the search region are returned.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("unselect_all_cells", &db::RecursiveShapeIterator::unselect_all_cells, "@brief Unselects all cells.\n" "\n" "This method will set the \"unselected\" mark on all cells. The effect is " "that subsequent calls of \\select_cells will select only the specified cells, not " "their children, because they are still unselected.\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("select_all_cells", &db::RecursiveShapeIterator::select_all_cells, "@brief Selects all cells.\n" "\n" "This method will set the \"selected\" mark on all cells. The effect is " "that subsequent calls of \\unselect_cells will unselect only the specified cells, not " "their children, because they are still unselected.\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method_ext ("unselect_cells", &unselect_cells1, "@brief Unselects the given cells.\n" "@args cells\n" "\n" "This method will sets the \"unselected\" mark on the given cells. " "That means that these cells or their child cells will not be visited, unless " "they are marked as \"selected\" again with the \\select_cells method.\n" "\n" "The cells are given as a list of cell indexes.\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method_ext ("unselect_cells", &unselect_cells2, "@brief Unselects the given cells.\n" "@args cells\n" "\n" "This method will sets the \"unselected\" mark on the given cells. " "That means that these cells or their child cells will not be visited, unless " "they are marked as \"selected\" again with the \\select_cells method.\n" "\n" "The cells are given as a glob pattern.\n" "A glob pattern follows the syntax of " "file names on the shell (i.e. \"A*\" are all cells starting with a letter \"A\").\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method_ext ("select_cells", &select_cells1, "@brief Unselects the given cells.\n" "@args cells\n" "\n" "This method will sets the \"selected\" mark on the given cells. " "That means that these cells or their child cells are visited, unless " "they are marked as \"unselected\" again with the \\unselect_cells method.\n" "\n" "The cells are given as a list of cell indexes.\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method_ext ("select_cells", &select_cells2, "@brief Unselects the given cells.\n" "@args cells\n" "\n" "This method will sets the \"selected\" mark on the given cells. " "That means that these cells or their child cells are visited, unless " "they are marked as \"unselected\" again with the \\unselect_cells method.\n" "\n" "The cells are given as a glob pattern.\n" "A glob pattern follows the syntax of " "file names on the shell (i.e. \"A*\" are all cells starting with a letter \"A\").\n" "\n" "This method will also reset the iterator.\n" "\n" "This method has been introduced in version 0.23.\n" ) + gsi::method ("shape_flags=", &db::RecursiveShapeIterator::shape_flags, "@brief Specifies the shape selection flags\n" "@args flags\n" "\n" "The flags are the same then being defined in \\Shapes (the default is RBA::Shapes::SAll).\n" "The flags must be specified before the shapes are being retrieved.\n" "Settings the shapes flags will reset the iterator.\n" ) + gsi::method ("trans", &db::RecursiveShapeIterator::trans, "@brief Gets the current transformation by which the shapes must be transformed into the initial cell\n" "\n" "The shapes delivered are not transformed. Instead, this transformation must be applied to \n" "get the shape in the coordinate system of the top cell.\n" "\n" "Starting with version 0.25, this transformation is a int-to-int transformation and there " "no longer is a 'itrans' method." ) + gsi::method ("shape", &db::RecursiveShapeIterator::shape, "@brief Gets the current shape\n" "\n" "Returns the shape currently referred to by the recursive iterator. \n" "This shape is not transformed yet and is located in the current cell.\n" ) + gsi::method ("at_end?", &db::RecursiveShapeIterator::at_end, "@brief End of iterator predicate\n" "\n" "Returns true, if the iterator is at the end of the sequence\n" ) + gsi::method ("cell", &db::RecursiveShapeIterator::cell, "@brief Gets the current cell's object \n" "\n" "This method has been introduced in version 0.23." ) + gsi::method ("cell_index", &db::RecursiveShapeIterator::cell_index, "@brief Gets the current cell's index \n" ) + gsi::method ("next", &db::RecursiveShapeIterator::next, "@brief Increment the iterator\n" "This moves the iterator to the next shape inside the search scope." ) + gsi::method ("layer", &db::RecursiveShapeIterator::layer, "@brief Returns the layer index where the current shape is coming from.\n" "\n" "This method has been introduced in version 0.23." ) + gsi::method ("path", &db::RecursiveShapeIterator::path, "@brief Gets the instantatiation path of the shape addressed currently\n" "\n" "This attribute is a sequence of \\InstElement objects describing the cell instance path from the initial " "cell to the current cell containing the current shape.\n" "\n" "This method has been introduced in version 0.25." ) + gsi::method ("==", &db::RecursiveShapeIterator::operator==, "@brief Comparison of iterators - equality\n" "@args other\n" "\n" "Two iterators are equal if they point to the same shape.\n" ) + gsi::method ("!=", &db::RecursiveShapeIterator::operator!=, "@brief Comparison of iterators - inequality\n" "@args other\n" "\n" "Two iterators are not equal if they do not point to the same shape.\n" ), "@brief An iterator delivering shapes that touch or overlap the given region recursively\n" "\n" "The iterator can be obtained from a layout, specifying a starting cell, a layer and optionally a region.\n" "It simplifies retrieval of shapes from a geometrical region while considering\n" "subcells as well.\n" "Some options can be specified, i.e. the level to which to look into or\n" "shape classes and shape properties. The shapes are retrieved by using the \\shape method,\n" "\\next moves to the next shape and \\at_end tells, if the iterator has move shapes to deliver.\n" "\n" "This is some sample code:\n" "\n" "@code\n" "# print the polygon-like objects as seen from the initial cell \"cell\"\n" "iter = layout.begin_shapes(cell_index, layer)\n" "while !iter.at_end?\n" " if iter.shape.renders_polygon?\n" " polygon = iter.shape.polygon.transformed(iter.itrans)\n" " puts \"In cell #{iter.cell.name}: \" + polyon.to_s\n" " end\n" " iter.next\n" "end\n" "@/code\n" "\n" "\\Layout offers three methods to get these iterators: begin_shapes, begin_shapes_touching and begin_shapes_overlapping.\n" "\\Layout#begin_shapes will deliver a standard recursive shape iterator which starts from the given cell and iterates " "over all child cells. \\Layout#begin_shapes_touching delivers a RecursiveShapeIterator which delivers the shapes " "whose bounding boxed touch the given search box. \\Layout#begin_shapes_overlapping delivers all shapes whose bounding box " "overlaps the search box.\n" "\n" "A RecursiveShapeIterator object can also be created explicitly. This allows some more options, i.e. using " "multiple layers. A multi-layer recursive shape iterator can be created like this:\n" "\n" "@code\n" "iter = RBA::RecursiveShapeIterator::new(layout, cell, [ layer_index1, layer_index2 .. ])\n" "@/code\n" "\n" "\"layout\" is the layout object, \"cell\" the RBA::Cell object of the initial cell. layer_index1 etc. are the " "layer indexes of the layers to get the shapes from. While iterating, \\RecursiveShapeIterator#layer delivers " "the layer index of the current shape.\n" "\n" "The recursive shape iterator can be confined to a maximum hierarchy depth. By using \\max_depth=, the " "iterator will restrict the search depth to the given depth in the cell tree.\n" "\n" "In addition, the recursive shape iterator supports selection and exclusion of subtrees. For that purpose " "it keeps flags per cell telling it for which cells to turn shape delivery on and off. The \\select_cells method " "sets the \"start delivery\" flag while \\unselect_cells sets the \"stop delivery\" flag. In effect, using " "\\unselect_cells will exclude that cell plus the subtree from delivery. Parts of that subtree can be " "turned on again using \\select_cells. For the cells selected that way, the shapes of these cells and their " "child cells are delivered, even if their parents was unselected.\n" "\n" "To get shapes from a specific cell, i.e. \"MACRO\" plus its child cells, unselect the top cell first " "and the select the desired cell again:\n" "\n" "@code\n" "# deliver all shapes inside \"MACRO\" and the sub-hierarchy:\n" "iter = RBA::RecursiveShapeIterator::new(layout, cell, layer)\n" "iter.unselect_cells(cell.cell_index)\n" "iter.select_cells(\"MACRO\")\n" "@/code\n" "\n" "Note that if \"MACRO\" uses library cells for example which are used otherwise as well, the " "iterator will only deliver the shapes for those instances belonging to \"MACRO\" (directly or indirectly), " "not those for other instances of these library cells.\n" "\n" "The \\unselect_all_cells and \\select_all_cells methods turn on the \"stop\" and \"start\" flag " "for all cells respectively. If you use \\unselect_all_cells and use \\select_cells for a specific cell, " "the iterator will deliver only the shapes of the selected cell, not its children. Those are still " "unselected by \\unselect_all_cells:\n" "\n" "@code\n" "# deliver all shapes of \"MACRO\" but not of child cells:\n" "iter = RBA::RecursiveShapeIterator::new(layout, cell, layer)\n" "iter.unselect_all_cells\n" "iter.select_cells(\"MACRO\")\n" "@/code\n" "\n" "Cell selection is done using cell indexes or glob pattern. Glob pattern are equivalent to the usual " "file name wildcards used on various command line shells. For example \"A*\" matches all cells starting with " "an \"A\". The curly brace notation and character classes are supported as well. For example \"C{125,512}\" matches " "\"C125\" and \"C512\" and \"[ABC]*\" matches all cells starting with an \"A\", a \"B\" or \"C\". \"[^ABC]*\" matches " "all cells not starting with one of that letters.\n" "\n" "The RecursiveShapeIterator class has been introduced in version 0.18 and has been extended substantially in 0.23.\n" ); }