summaryrefslogtreecommitdiff
path: root/hyp2mat/lib/hyperlynx.cc
diff options
context:
space:
mode:
Diffstat (limited to 'hyp2mat/lib/hyperlynx.cc')
-rw-r--r--hyp2mat/lib/hyperlynx.cc348
1 files changed, 348 insertions, 0 deletions
diff --git a/hyp2mat/lib/hyperlynx.cc b/hyp2mat/lib/hyperlynx.cc
new file mode 100644
index 0000000..adaa043
--- /dev/null
+++ b/hyp2mat/lib/hyperlynx.cc
@@ -0,0 +1,348 @@
+/*
+ * hyp2mat - convert hyperlynx files to matlab scripts
+ * Copyright 2012 Koen De Vleeschauwer.
+ *
+ * This file is part of hyp2mat.
+ *
+ * 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 3 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, see <http://www.gnu.org/licenses/>.
+ */
+
+/*
+ * HypFile is concerned with parsing the hyperlynx file, and contains many variables and methods
+ * which are only of interest during parsing.
+ *
+ * Hyp2Mat is interested in presenting an uncluttered interface.
+ *
+ * This is where data is passed on from HypFile to Hyp2Mat, from the obscure to the obvious.
+ */
+
+#include <iostream>
+#include <algorithm>
+
+#include "config.h"
+#include "hyperlynx.h"
+#include "polygon.h"
+#include "crop.h"
+
+HyperLynx::HyperLynx ()
+{
+ raw = false;
+ arc_precision = 0;
+ clearance = 0.0002;
+}
+
+
+/*
+ * Read a pcb in hyperlynx format
+ */
+
+void HyperLynx::Read (std::string filename, Hyp2Mat::PCB& pcb)
+{
+ /*
+ * load hyperlynx file in hyp_file
+ */
+
+ bool success = LoadHypFile(pcb, filename, arc_precision);
+ if (!success) throw ("load hyp file error");
+
+ /*
+ * Copy hyperlynx file data from hyp_file to pcb
+ */
+ saved_bounds = bounds;
+ CopyBoard(pcb); /* copy board outline. crop board in x, y axis */
+
+ CopyStackUp(pcb); /* copy layer stackup */
+ bounds = AdjustBounds(pcb, saved_bounds);
+ CopyVias(pcb); /* copy vias */
+ CopyViaPlating(pcb); /* copy via plating thickness */
+ CopyDevices(pcb); /* copy device info */
+ CopyPins(pcb); /* copy pin references */
+
+ bounds = AdjustBounds(pcb, saved_bounds);
+ CropLayers(pcb, bounds); /* crop board in z axis */
+
+ if (bounds.z_min == bounds.z_max) std::cerr << "warning: zero height board" << std::endl;
+
+ /*
+ * Check whether to print all net names
+ */
+
+ /* if one of the nets is "?", print list of all nets */
+ if (std::find(nets.begin(), nets.end(), "?") != nets.end())
+ PrintNets();
+
+ return;
+}
+
+/*
+ * Load Hyperlynx file in hyp_file
+ */
+
+bool HyperLynx::LoadHypFile(Hyp2Mat::PCB& pcb, std::string filename, double arc_precision)
+{
+ hyp_file.trace_scanner = (pcb.debug == 4);
+ hyp_file.trace_parser = (pcb.debug == 3);
+ hyp_file.trace_hyp = (pcb.debug == 2);
+
+ if (pcb.debug != 0) {
+ std::cerr << "hyp2mat " << VERSION << " using ClipperLib " << CLIPPER_VERSION << std::endl;
+ }
+
+ if (arc_precision < 0) {
+ std::cerr << "arc precision negative" << std::endl;
+ arc_precision = 0;
+ }
+ hyp_file.arc_precision = arc_precision;
+
+ hyp_file.flood_layers = flood_layers;
+
+ bool success = hyp_file.load(filename);
+
+ if (pcb.debug == 1) {
+ std::cerr << "writing hyp_file dump to hyp_debug.txt" << std::endl;
+ hyp_file.save ("hyp_debug.txt");
+ }
+
+ return success;
+}
+
+/*
+ * copy board outline into polygon "board" and crop according to "bounds".
+ */
+
+void HyperLynx::CopyBoard(Hyp2Mat::PCB& pcb)
+{
+ bool outer_edge = true;
+
+ for (HypFile::PolygonList::iterator i = hyp_file.board.edge.begin(); i != hyp_file.board.edge.end(); ++i) {
+ Hyp2Mat::Polygon poly;
+ Hyp2Mat::FloatPolygon edge;
+ for (HypFile::PointList::iterator j = i->vertex.begin(); j != i->vertex.end(); ++j)
+ edge.push_back(Hyp2Mat::FloatPoint(j->x, j->y));
+ poly.AddEdge(edge);
+ if (outer_edge)
+ board.Union(poly);
+ else
+ board.Difference(poly);
+ outer_edge = false;
+ }
+
+ /*
+ * Cropping - Remove board outside bounds
+ */
+
+ pcb.board = board.Result(); /* store intermediate result */
+ bounds = AdjustBounds(pcb, bounds); /* tighten bounds around board */
+ board = CropPolygon(board, bounds); /* crop board. store board as integer polygon */
+ pcb.board = board.Result(); /* board as floating point polygon */
+
+ return;
+}
+
+/*
+ * copy vias
+ */
+
+void HyperLynx::CopyVias(Hyp2Mat::PCB& pcb)
+{
+ for (HypFile::NetList::iterator i = hyp_file.net.begin(); i != hyp_file.net.end(); ++i) {
+
+ /* check if we're interested in this net. if no nets are specified, copy all nets */
+ bool net_wanted = nets.empty() || (std::find(nets.begin(), nets.end(), i->net_name) != nets.end());
+ if (!net_wanted) continue;
+
+ /* copy vias of current net */
+ for (HypFile::ViaList::iterator j = i->via.begin(); j != i->via.end(); ++j) {
+ Hyp2Mat::Via v;
+ v.x = j->x;
+ v.y = j->y;
+ v.z0 = j->z0;
+ v.z1 = j->z1;
+ v.radius = j->radius;
+
+ /* cropping */
+ if (!board.IsEmpty() && ((v.x > bounds.x_max) || (v.x < bounds.x_min) || (v.y > bounds.y_max) || (v.y < bounds.y_min))) continue; // Only crop in x and y if board outline is defined
+ if ((v.z0 > bounds.z_max) && (v.z1 > bounds.z_max)) continue;
+ if ((v.z0 < bounds.z_min) && (v.z1 < bounds.z_min)) continue;
+ if (v.z1 > bounds.z_max) v.z1 = bounds.z_max;
+ if (v.z0 < bounds.z_min) v.z0 = bounds.z_min;
+
+ /* add to list of vias */
+ pcb.via.push_back(v);
+ }
+ }
+
+ return;
+}
+
+/*
+ * copy via plating
+ */
+
+void HyperLynx::CopyViaPlating(Hyp2Mat::PCB& pcb)
+{
+ /* assume via plating thickness is identical to outer plating thickness */
+ for (HypFile::LayerList::iterator l = hyp_file.stackup.begin(); l != hyp_file.stackup.end(); ++l)
+ if (l->layer_type != HypFile::LAYER_DIELECTRIC) {
+ /* first metal layer */
+ pcb.via_plating_thickness = l->plating_thickness;
+ break;
+ }
+
+ return;
+}
+
+/*
+ * copy devices
+ */
+
+void HyperLynx::CopyDevices(Hyp2Mat::PCB& pcb)
+{
+ for (HypFile::DeviceList::iterator i = hyp_file.device.begin(); i != hyp_file.device.end(); ++i) {
+ /* set up device */
+ Hyp2Mat::Device dev;
+ dev.ref = i->ref;
+ dev.name = i->name;
+ dev.layer_name = i->layer_name;
+
+ if (i->value_float_set) {
+ dev.value_float = i->value_float;
+ dev.value_type = Hyp2Mat::DEVICE_VALUE_FLOAT;
+ }
+ else if (i->value_string_set) {
+ dev.value_string = i->value_string;
+ dev.value_type = Hyp2Mat::DEVICE_VALUE_STRING;
+ }
+ else dev.value_type = Hyp2Mat::DEVICE_VALUE_NONE;
+
+ /* add device to board */
+ pcb.device.push_back(dev);
+ }
+ return;
+}
+
+/*
+ * copy pins
+ */
+
+void HyperLynx::CopyPins(Hyp2Mat::PCB& pcb)
+{
+ /* iterate over all pins */
+ for (HypFile::NetList::iterator i = hyp_file.net.begin(); i != hyp_file.net.end(); ++i) {
+
+ /* check if we're interested in this net. if no nets are specified, copy all nets */
+ bool net_wanted = nets.empty() || (std::find(nets.begin(), nets.end(), i->net_name) != nets.end());
+ if (!net_wanted) continue;
+
+ for (HypFile::PinList::iterator j = i->pin.begin(); j != i->pin.end(); ++j)
+ /* find pin padstack */
+ for (HypFile::PadstackList::iterator p = hyp_file.padstack.begin(); p != hyp_file.padstack.end(); ++p)
+ if (p->padstack_name == j->padstack_name)
+ /* copy this pin */
+ CopyPin(pcb, *j, *p);
+ }
+
+ return;
+}
+
+/*
+ * copy a single pin
+ */
+
+void HyperLynx::CopyPin(Hyp2Mat::PCB& pcb, HypFile::Pin& pin, HypFile::Padstack& padstack)
+{
+ Hyp2Mat::Pin new_pin;
+ new_pin.x = Hyp2Mat::Polygon::AlignToGrid(pin.x);
+ new_pin.y = Hyp2Mat::Polygon::AlignToGrid(pin.y);
+ new_pin.ref = pin.pin_reference;
+
+ HypFile::Pad pad;
+
+ std::string layer_name;
+
+ /* now find pad */
+
+ /* how many layers does the padstack have? */
+ if (padstack.pads.size() == 1) {
+ /* only a single layer: that's easy */
+ pad = padstack.pads.front();
+ }
+ else {
+ /* determine pin device name */
+ size_t found = pin.pin_reference.find('.');
+ if ((found == std::string::npos) || (found == 0)) return; /* device not found */
+ std::string device_name = pin.pin_reference.substr(0, found);
+
+ /* find device pin belongs to */
+ bool device_found = false;
+ for (Hyp2Mat::DeviceList::iterator i = pcb.device.begin(); i != pcb.device.end(); ++i)
+ if (device_name == i->ref) {
+ device_found = true;
+ layer_name = i->layer_name; /* device layer */
+ }
+ if (!device_found) return;
+
+ /* we now know the device layer. Find this layer in the pin padstack. */
+ bool pad_found = false;
+ for (HypFile::PadList::iterator p = padstack.pads.begin(); p != padstack.pads.end(); ++p)
+ if (p->layer_name == layer_name) {
+ pad_found = true;
+ pad = *p;
+ }
+ if (!pad_found) return;
+ }
+
+ /* find layer vertical position */
+ for (Hyp2Mat::LayerList::iterator l = pcb.stackup.begin(); l != pcb.stackup.end(); ++l)
+ if (pad.layer_name == l->layer_name) {
+ new_pin.layer_name = l->layer_name;
+ new_pin.z0 = l->z0;
+ new_pin.z1 = l->z1;
+ }
+
+ /* convert pad to polygon, and copy pad shape */
+ HypFile::Polygon pad_poly = hyp_file.pad2poly(new_pin.x, new_pin.y, pad);
+ new_pin.metal.clear();
+ for (HypFile::PointList::iterator j = pad_poly.vertex.begin(); j != pad_poly.vertex.end(); ++j)
+ new_pin.metal.push_back(Hyp2Mat::FloatPoint(Hyp2Mat::Polygon::AlignToGrid(j->x), Hyp2Mat::Polygon::AlignToGrid(j->y)));
+
+ /* add pin to pcb */
+ pcb.pin.push_back(new_pin);
+
+ return;
+}
+
+/*
+ * Print all available nets
+ */
+
+void HyperLynx::PrintNets()
+{
+ std::vector<std::string> net_names;
+
+ for (HypFile::NetList::iterator i = hyp_file.net.begin(); i != hyp_file.net.end(); ++i)
+ net_names.push_back(i->net_name);
+
+ sort(net_names.begin(), net_names.end());
+ unique(net_names.begin(), net_names.end());
+
+ std::cout << "nets:";
+ for (std::vector<std::string>::iterator i = net_names.begin(); i != net_names.end(); ++i)
+ std::cout << " " << *i;
+ std::cout << std::endl;
+
+ return;
+}
+
+/* not truncated */