/*
* 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 .
*/
/*
* 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
#include
#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 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::iterator i = net_names.begin(); i != net_names.end(); ++i)
std::cout << " " << *i;
std::cout << std::endl;
return;
}
/* not truncated */