klayout/src/db/db/dbRegion.cc

737 lines
21 KiB
C++

/*
KLayout Layout Viewer
Copyright (C) 2006-2024 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 "dbRegion.h"
#include "dbRegionUtils.h"
#include "dbOriginalLayerRegion.h"
#include "dbEmptyRegion.h"
#include "dbFlatRegion.h"
#include "dbDeepRegion.h"
#include "dbDeepEdges.h"
#include "dbFlatEdges.h"
#include "dbPolygonTools.h"
#include "dbCompoundOperation.h"
#include "dbLayout.h"
#include "dbWriter.h"
#include "tlStream.h"
#include "tlGlobPattern.h"
// NOTE: include this to provide the symbols for "make_variant"
#include "gsiDecl.h"
namespace db
{
// -------------------------------------------------------------------------------------------------------------
// Region implementation
Region::Region ()
: mp_delegate (new EmptyRegion ())
{
// .. nothing yet ..
}
Region::Region (RegionDelegate *delegate)
: mp_delegate (delegate)
{
// .. nothing yet ..
}
Region::Region (const Region &other)
: db::ShapeCollection (), mp_delegate (other.mp_delegate->clone ())
{
// .. nothing yet ..
}
Region::~Region ()
{
delete mp_delegate;
mp_delegate = 0;
}
Region &Region::operator= (const Region &other)
{
if (this != &other) {
set_delegate (other.mp_delegate->clone (), false);
}
return *this;
}
Region::Region (const RecursiveShapeIterator &si, bool merged_semantics, bool is_merged)
{
mp_delegate = new OriginalLayerRegion (si, db::ICplxTrans (), merged_semantics, is_merged);
}
Region::Region (const RecursiveShapeIterator &si, const db::ICplxTrans &trans, bool merged_semantics, bool is_merged)
{
mp_delegate = new OriginalLayerRegion (si, trans, merged_semantics, is_merged);
}
Region::Region (const Shapes &shapes, bool merged_semantics, bool is_merged)
{
db::FlatRegion *flat_region = new FlatRegion (is_merged);
flat_region->reserve (shapes.size (db::ShapeIterator::Regions));
// NOTE: we need to normalize the shapes to polygons because this is what the flat region expects
for (auto s = shapes.begin (db::ShapeIterator::Regions); ! s.at_end (); ++s) {
flat_region->insert (*s);
}
mp_delegate = flat_region;
mp_delegate->set_merged_semantics (merged_semantics);
}
Region::Region (const Shapes &shapes, const db::ICplxTrans &trans, bool merged_semantics, bool is_merged)
{
db::FlatRegion *flat_region = new FlatRegion (is_merged);
flat_region->reserve (shapes.size (db::ShapeIterator::Regions));
// NOTE: we need to normalize the shapes to polygons because this is what the flat region expects
for (auto s = shapes.begin (db::ShapeIterator::Regions); ! s.at_end (); ++s) {
flat_region->insert (*s, trans);
}
mp_delegate = flat_region;
mp_delegate->set_merged_semantics (merged_semantics);
}
Region::Region (const RecursiveShapeIterator &si, DeepShapeStore &dss, double area_ratio, size_t max_vertex_count)
{
mp_delegate = new DeepRegion (si, dss, area_ratio, max_vertex_count);
}
Region::Region (const RecursiveShapeIterator &si, DeepShapeStore &dss, const db::ICplxTrans &trans, bool merged_semantics, double area_ratio, size_t max_vertex_count)
{
mp_delegate = new DeepRegion (si, dss, trans, merged_semantics, area_ratio, max_vertex_count);
}
Region::Region (DeepShapeStore &dss)
{
tl_assert (dss.is_singular ());
unsigned int layout_index = 0; // singular layout index
mp_delegate = new db::DeepRegion (db::DeepLayer (&dss, layout_index, dss.layout (layout_index).insert_layer ()));
}
void
Region::write (const std::string &fn) const
{
// method provided for debugging purposes
db::Layout layout;
const db::Cell &top = layout.cell (layout.add_cell ("REGION"));
unsigned int li = layout.insert_layer (db::LayerProperties (0, 0));
insert_into (&layout, top.cell_index (), li);
tl::OutputStream os (fn);
db::SaveLayoutOptions opt;
opt.set_format_from_filename (fn);
db::Writer writer (opt);
writer.write (layout, os);
}
const db::RecursiveShapeIterator &
Region::iter () const
{
static db::RecursiveShapeIterator def_iter;
const db::RecursiveShapeIterator *i = mp_delegate ? mp_delegate->iter () : 0;
return *(i ? i : &def_iter);
}
void
Region::set_delegate (RegionDelegate *delegate, bool keep_attributes)
{
if (delegate != mp_delegate) {
if (keep_attributes && delegate && mp_delegate) {
// copy the basic attributes like #threads etc.
delegate->RegionDelegate::operator= (*mp_delegate);
}
delete mp_delegate;
mp_delegate = delegate;
}
}
void
Region::clear ()
{
set_delegate (new EmptyRegion ());
}
void
Region::reserve (size_t n)
{
mutable_region ()->reserve (n);
}
template <class T>
Region &Region::transform (const T &trans)
{
mutable_region ()->transform (trans);
return *this;
}
// explicit instantiations
template DB_PUBLIC Region &Region::transform (const db::ICplxTrans &);
template DB_PUBLIC Region &Region::transform (const db::Trans &);
template DB_PUBLIC Region &Region::transform (const db::Disp &);
template DB_PUBLIC Region &Region::transform (const db::IMatrix2d &);
template DB_PUBLIC Region &Region::transform (const db::IMatrix3d &);
template <class Sh>
void Region::insert (const Sh &shape)
{
mutable_region ()->insert (shape);
}
template DB_PUBLIC void Region::insert (const db::Box &);
template DB_PUBLIC void Region::insert (const db::BoxWithProperties &);
template DB_PUBLIC void Region::insert (const db::SimplePolygon &);
template DB_PUBLIC void Region::insert (const db::SimplePolygonWithProperties &);
template DB_PUBLIC void Region::insert (const db::Polygon &);
template DB_PUBLIC void Region::insert (const db::PolygonWithProperties &);
template DB_PUBLIC void Region::insert (const db::Path &);
template DB_PUBLIC void Region::insert (const db::PathWithProperties &);
void Region::insert (const db::Shape &shape)
{
mutable_region ()->insert (shape);
}
template <class T>
void Region::insert (const db::Shape &shape, const T &trans)
{
mutable_region ()->insert (shape, trans);
}
template DB_PUBLIC void Region::insert (const db::Shape &, const db::ICplxTrans &);
template DB_PUBLIC void Region::insert (const db::Shape &, const db::Trans &);
template DB_PUBLIC void Region::insert (const db::Shape &, const db::Disp &);
MutableRegion *
Region::mutable_region ()
{
MutableRegion *region = dynamic_cast<MutableRegion *> (mp_delegate);
if (! region) {
FlatRegion *flat_region = new FlatRegion ();
region = flat_region;
if (mp_delegate) {
flat_region->RegionDelegate::operator= (*mp_delegate); // copy basic flags
flat_region->insert_seq (begin ());
flat_region->set_is_merged (mp_delegate->is_merged ());
}
set_delegate (flat_region);
}
return region;
}
EdgePairs
Region::cop_to_edge_pairs (db::CompoundRegionOperationNode &node, db::PropertyConstraint prop_constraint)
{
tl_assert (node.result_type () == db::CompoundRegionOperationNode::EdgePairs);
return EdgePairs (mp_delegate->cop_to_edge_pairs (node, prop_constraint));
}
Region
Region::cop_to_region (db::CompoundRegionOperationNode &node, db::PropertyConstraint prop_constraint)
{
tl_assert (node.result_type () == db::CompoundRegionOperationNode::Region);
return Region (mp_delegate->cop_to_region (node, prop_constraint));
}
Edges
Region::cop_to_edges (db::CompoundRegionOperationNode &node, db::PropertyConstraint prop_constraint)
{
tl_assert (node.result_type () == db::CompoundRegionOperationNode::Edges);
return Edges (mp_delegate->cop_to_edges (node, prop_constraint));
}
tl::Variant
Region::cop (db::CompoundRegionOperationNode &node, db::PropertyConstraint prop_constraint)
{
if (node.result_type () == db::CompoundRegionOperationNode::EdgePairs) {
return tl::Variant::make_variant (new EdgePairs (mp_delegate->cop_to_edge_pairs (node, prop_constraint)));
} else if (node.result_type () == db::CompoundRegionOperationNode::Edges) {
return tl::Variant::make_variant (new Edges (mp_delegate->cop_to_edges (node, prop_constraint)));
} else if (node.result_type () == db::CompoundRegionOperationNode::Region) {
return tl::Variant::make_variant (new Region (mp_delegate->cop_to_region (node, prop_constraint)));
} else {
return tl::Variant ();
}
}
Region &
Region::size (coord_type d, unsigned int mode)
{
set_delegate (mp_delegate->sized (d, mode));
return *this;
}
Region &
Region::size (coord_type dx, coord_type dy, unsigned int mode)
{
set_delegate (mp_delegate->sized (dx, dy, mode));
return *this;
}
Region
Region::sized (coord_type d, unsigned int mode) const
{
return Region (mp_delegate->sized (d, mode));
}
Region
Region::sized (coord_type dx, coord_type dy, unsigned int mode) const
{
return Region (mp_delegate->sized (dx, dy, mode));
}
Region &
Region::size_inside (const db::Region &inside, bool outside, coord_type d, int steps, unsigned int mode)
{
set_delegate (mp_delegate->sized_inside (inside, outside, d, steps, mode));
return *this;
}
Region &
Region::size_inside (const db::Region &inside, bool outside, coord_type dx, coord_type dy, int steps, unsigned int mode)
{
set_delegate (mp_delegate->sized_inside (inside, outside, dx, dy, steps, mode));
return *this;
}
Region
Region::sized_inside (const db::Region &inside, bool outside, coord_type d, int steps, unsigned int mode) const
{
return Region (mp_delegate->sized_inside (inside, outside, d, steps, mode));
}
Region
Region::sized_inside (const db::Region &inside, bool outside, coord_type dx, coord_type dy, int steps, unsigned int mode) const
{
return Region (mp_delegate->sized_inside (inside, outside, dx, dy, steps, mode));
}
void
Region::round_corners (double rinner, double router, unsigned int n)
{
process (RoundedCornersProcessor (rinner, router, n));
}
Region
Region::rounded_corners (double rinner, double router, unsigned int n) const
{
return processed (RoundedCornersProcessor (rinner, router, n));
}
void
Region::smooth (coord_type d, bool keep_hv)
{
process (SmoothingProcessor (d, keep_hv));
}
Region
Region::smoothed (coord_type d, bool keep_hv) const
{
return processed (SmoothingProcessor (d, keep_hv));
}
db::Region &
Region::flatten ()
{
mutable_region ()->flatten ();
return *this;
}
void
Region::snap (db::Coord gx, db::Coord gy)
{
set_delegate (mp_delegate->snapped_in_place (gx, gy));
}
Region
Region::snapped (db::Coord gx, db::Coord gy) const
{
return Region (mp_delegate->snapped (gx, gy));
}
void
Region::scale_and_snap (db::Coord gx, db::Coord mx, db::Coord dx, db::Coord gy, db::Coord my, db::Coord dy)
{
set_delegate (mp_delegate->scaled_and_snapped_in_place (gx, mx, dx, gy, my, dy));
}
Region
Region::scaled_and_snapped (db::Coord gx, db::Coord mx, db::Coord dx, db::Coord gy, db::Coord my, db::Coord dy) const
{
return Region (mp_delegate->scaled_and_snapped (gx, mx, dx, gy, my, dy));
}
Region
Region::strange_polygon_check () const
{
return Region (processed (StrangePolygonCheckProcessor ()));
}
Region
Region::holes () const
{
return Region (processed (HolesExtractionProcessor ()));
}
Region
Region::hulls () const
{
return Region (processed (HullExtractionProcessor ()));
}
namespace
{
template <class Container>
struct dot_delivery
{
typedef Container container_type;
dot_delivery ()
{
// .. nothing yet ..
}
void insert (const db::Point &pt, Container *container) const
{
container->insert (db::Edge (pt, pt));
}
};
template <class Container>
struct box_delivery
{
typedef Container container_type;
box_delivery (db::Coord enl)
: m_d (enl, enl)
{
// .. nothing yet ..
}
void insert (const db::Point &pt, Container *container) const
{
container->insert (db::Box (pt - m_d, pt + m_d));
}
private:
db::Vector m_d;
};
template <class Iter, class Delivery>
static void fill_texts (const Iter &iter, const std::string &pat, bool pattern, const Delivery &delivery, typename Delivery::container_type *container, const db::ICplxTrans &trans, const db::DeepRegion *org_deep)
{
std::pair<bool, db::property_names_id_type> text_annot_name_id;
const db::Layout *layout = 0;
if (org_deep) {
// NOTE: deep regions can store texts in a special way - as small boxes with a special property attached.
// The property will give the text string. This function can restore these pseudo-texts as Text objects.
layout = &org_deep->deep_layer ().layout ();
const db::DeepShapeStore *store = org_deep->deep_layer ().store ();
if (! store->text_property_name ().is_nil ()) {
text_annot_name_id = db::PropertiesRepository::instance ().get_id_of_name (store->text_property_name ());
}
}
tl::GlobPattern glob_pat;
bool all = false;
if (pattern) {
if (pat == "*") {
all = true;
} else {
glob_pat = tl::GlobPattern (pat);
}
}
for (Iter si = iter; ! si.at_end (); ++si) {
bool is_text = false;
std::string text_string;
if (si->is_text ()) {
// a raw text
is_text = true;
text_string = si->text_string ();
} else if (layout && text_annot_name_id.first && si->prop_id () > 0) {
// a text marker
const db::PropertiesSet &ps = db::properties (si->prop_id ());
for (db::PropertiesSet::iterator j = ps.begin (); j != ps.end () && ! is_text; ++j) {
if (j->first == text_annot_name_id.second) {
text_string = db::property_value (j->second).to_string ();
is_text = true;
}
}
}
if (is_text &&
(all || (pattern && glob_pat.match (text_string)) || (!pattern && text_string == pat))) {
delivery.insert (si.trans () * (trans * si->bbox ().center ()), container);
}
}
}
template <class Delivery>
class text_shape_receiver
: public db::HierarchyBuilderShapeReceiver
{
public:
text_shape_receiver (const Delivery &delivery, const std::string &pat, bool pattern, const db::DeepRegion *org_deep)
: m_delivery (delivery), m_glob_pat (), m_all (false), m_pattern (pattern), m_pat (pat), m_text_annot_name_id (false, 0), mp_layout (0)
{
if (org_deep) {
mp_layout = & org_deep->deep_layer ().layout ();
const db::DeepShapeStore *store = org_deep->deep_layer ().store ();
if (! store->text_property_name ().is_nil ()) {
m_text_annot_name_id = db::PropertiesRepository::instance ().get_id_of_name (store->text_property_name ());
}
}
if (pattern) {
if (m_pat == "*") {
m_all = true;
} else {
m_glob_pat = tl::GlobPattern (pat);
}
}
}
virtual void push (const db::Shape &shape, db::properties_id_type, const db::ICplxTrans &trans, const db::Box &region, const db::RecursiveShapeReceiver::box_tree_type *complex_region, db::Shapes *target)
{
bool is_text = false;
std::string text_string;
if (shape.is_text ()) {
// a raw text
is_text = true;
text_string = shape.text_string ();
} else if (mp_layout && m_text_annot_name_id.first && shape.prop_id () > 0) {
// a text marker
const db::PropertiesSet &ps = db::properties (shape.prop_id ());
for (db::PropertiesSet::iterator j = ps.begin (); j != ps.end () && ! is_text; ++j) {
if (j->first == m_text_annot_name_id.second) {
text_string = db::property_value (j->second).to_string ();
is_text = true;
}
}
}
if (is_text &&
(m_all || (m_pattern && m_glob_pat.match (text_string)) || (!m_pattern && text_string == m_pat))) {
db::Point pt = shape.bbox ().center ();
if (! complex_region) {
if (region.contains (pt)) {
m_delivery.insert (pt.transformed (trans), target);
}
} else {
if (! complex_region->begin_overlapping (db::Box (pt, pt), db::box_convert<db::Box> ()).at_end ()) {
m_delivery.insert (pt.transformed (trans), target);
}
}
}
}
virtual void push (const db::Box &, db::properties_id_type, const db::ICplxTrans &, const db::Box &, const db::RecursiveShapeReceiver::box_tree_type *, db::Shapes *) { }
virtual void push (const db::Polygon &, db::properties_id_type, const db::ICplxTrans &, const db::Box &, const db::RecursiveShapeReceiver::box_tree_type *, db::Shapes *) { }
private:
Delivery m_delivery;
tl::GlobPattern m_glob_pat;
bool m_all;
bool m_pattern;
std::string m_pat;
std::pair<bool, db::property_names_id_type> m_text_annot_name_id;
const db::Layout *mp_layout;
};
}
Edges
Region::texts_as_dots (const std::string &pat, bool pattern) const
{
const db::DeepRegion *dr = dynamic_cast<const db::DeepRegion *> (delegate ());
if (dr) {
return texts_as_dots (pat, pattern, const_cast<db::DeepShapeStore &> (*dr->deep_layer ().store ()));
}
std::pair<db::RecursiveShapeIterator, db::ICplxTrans> si = begin_iter ();
if (! dr) {
// some optimization
si.first.shape_flags (si.first.shape_flags () & db::ShapeIterator::Texts);
}
std::unique_ptr<db::FlatEdges> res (new db::FlatEdges ());
res->set_merged_semantics (false);
fill_texts (si.first, pat, pattern, dot_delivery<db::FlatEdges> (), res.get (), si.second, dr);
return Edges (res.release ());
}
Edges
Region::texts_as_dots (const std::string &pat, bool pattern, db::DeepShapeStore &store) const
{
const db::DeepRegion *dr = dynamic_cast<const db::DeepRegion *> (delegate ());
std::pair<db::RecursiveShapeIterator, db::ICplxTrans> si = begin_iter ();
if (! dr) {
// some optimization
si.first.shape_flags (si.first.shape_flags () & db::ShapeIterator::Texts);
}
if (! si.first.layout ()) {
// flat fallback if the source isn't a deep or original layer
std::unique_ptr<db::FlatEdges> res (new db::FlatEdges ());
res->set_merged_semantics (false);
fill_texts (si.first, pat, pattern, dot_delivery<db::FlatEdges> (), res.get (), si.second, dr);
Edges edges (res.release ());
edges.set_merged_semantics (false);
return edges;
}
db::Edges edges;
text_shape_receiver<dot_delivery<db::Shapes> > pipe = text_shape_receiver<dot_delivery<db::Shapes> > (dot_delivery<db::Shapes> (), pat, pattern, dr);
if (dr && dr->deep_layer ().store () == &store) {
edges = Edges (new db::DeepEdges (store.create_copy (dr->deep_layer (), &pipe)));
} else {
edges = Edges (new db::DeepEdges (store.create_custom_layer (si.first, &pipe, si.second)));
}
edges.set_merged_semantics (false);
return edges;
}
Region
Region::texts_as_boxes (const std::string &pat, bool pattern, db::Coord enl) const
{
const db::DeepRegion *dr = dynamic_cast<const db::DeepRegion *> (delegate ());
if (dr) {
return texts_as_boxes (pat, pattern, enl, const_cast<db::DeepShapeStore &> (*dr->deep_layer ().store ()));
}
std::pair<db::RecursiveShapeIterator, db::ICplxTrans> si = begin_iter ();
if (! dr) {
// some optimization
si.first.shape_flags (si.first.shape_flags () & db::ShapeIterator::Texts);
}
std::unique_ptr<db::FlatRegion> res (new db::FlatRegion ());
res->set_merged_semantics (false);
fill_texts (si.first, pat, pattern, box_delivery<db::FlatRegion> (enl), res.get (), si.second, dr);
return Region (res.release ());
}
Region
Region::texts_as_boxes (const std::string &pat, bool pattern, db::Coord enl, db::DeepShapeStore &store) const
{
const db::DeepRegion *dr = dynamic_cast<const db::DeepRegion *> (delegate ());
std::pair<db::RecursiveShapeIterator, db::ICplxTrans> si = begin_iter ();
if (! dr) {
// some optimization
si.first.shape_flags (si.first.shape_flags () & db::ShapeIterator::Texts);
}
if (! si.first.layout ()) {
// flat fallback if the source isn't a deep or original layer
std::unique_ptr<db::FlatRegion> res (new db::FlatRegion ());
res->set_merged_semantics (false);
fill_texts (si.first, pat, pattern, box_delivery<db::FlatRegion> (enl), res.get (), si.second, dr);
return Region (res.release ());
}
text_shape_receiver<box_delivery<db::Shapes> > pipe = text_shape_receiver<box_delivery<db::Shapes> > (box_delivery<db::Shapes> (enl), pat, pattern, dr);
if (dr && dr->deep_layer ().store () == &store) {
return Region (new db::DeepRegion (store.create_copy (dr->deep_layer (), &pipe)));
} else {
return Region (new db::DeepRegion (store.create_custom_layer (si.first, &pipe, si.second)));
}
}
}
namespace tl
{
template<> DB_PUBLIC bool test_extractor_impl (tl::Extractor &ex, db::Region &b)
{
db::Polygon p;
if (ex.at_end ()) {
return true;
}
if (! ex.try_read (p)) {
return false;
}
b.insert (p);
while (ex.test (";")) {
ex.read (p);
b.insert (p);
}
return true;
}
template<> DB_PUBLIC void extractor_impl (tl::Extractor &ex, db::Region &b)
{
if (! test_extractor_impl (ex, b)) {
ex.error (tl::to_string (tr ("Expected a region specification")));
}
}
}