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

eu.mihosoft.freerouting.autoroute.InsertFoundConnectionAlgo Maven / Gradle / Ivy

The newest version!
/*
 *   Copyright (C) 2014  Alfons Wirtz
 *   website www.freerouting.net
 *
 *   Copyright (C) 2017 Michael Hoffer 
 *   Website www.freerouting.mihosoft.eu
*
 *   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 at  
 *   for more details.
 *
 * InsertFoundConnectionAlgo.java
 *
 * Created on 23. Februar 2004, 08:18
 */
package eu.mihosoft.freerouting.autoroute;

import eu.mihosoft.freerouting.geometry.planar.IntPoint;
import eu.mihosoft.freerouting.geometry.planar.Point;
import eu.mihosoft.freerouting.geometry.planar.FloatPoint;
import eu.mihosoft.freerouting.geometry.planar.Polyline;

import java.util.Iterator;
import java.util.Set;

import eu.mihosoft.freerouting.library.Padstack;
import eu.mihosoft.freerouting.rules.ViaInfo;

import eu.mihosoft.freerouting.board.ForcedViaAlgo;
import eu.mihosoft.freerouting.board.PolylineTrace;
import eu.mihosoft.freerouting.board.Trace;
import eu.mihosoft.freerouting.board.Item;
import eu.mihosoft.freerouting.board.RoutingBoard;
import eu.mihosoft.freerouting.board.ItemSelectionFilter;
import eu.mihosoft.freerouting.board.TestLevel;

/**
 * Inserts the traces and vias of the connection found by the eu.mihosoft.freerouting.autoroute algorithm.
 *
 * @author  Alfons Wirtz
 */
public class InsertFoundConnectionAlgo
{

    /**
     * Creates a new instance of InsertFoundConnectionAlgo .
     * Returns null, if the insertion did not succeed.
     */
    public static InsertFoundConnectionAlgo get_instance(LocateFoundConnectionAlgo p_connection,
            RoutingBoard p_board, AutorouteControl p_ctrl)
    {
        if (p_connection == null || p_connection.connection_items == null)
        {
            return null;
        }
        int curr_layer = p_connection.target_layer;
        InsertFoundConnectionAlgo new_instance = new InsertFoundConnectionAlgo(p_board, p_ctrl);
        Iterator it = p_connection.connection_items.iterator();
        while (it.hasNext())
        {
            LocateFoundConnectionAlgoAnyAngle.ResultItem curr_new_item = it.next();
            if (!new_instance.insert_via(curr_new_item.corners[0], curr_layer, curr_new_item.layer))
            {
                return null;
            }
            curr_layer = curr_new_item.layer;
            if (!new_instance.insert_trace(curr_new_item))
            {
                if (p_board.get_test_level().ordinal() >= TestLevel.CRITICAL_DEBUGGING_OUTPUT.ordinal())
                {
                    System.out.print("InsertFoundConnectionAlgo: insert trace failed for net ");
                    System.out.println(p_ctrl.net_no);
                }
                return null;
            }
        }
        if (!new_instance.insert_via(new_instance.last_corner, curr_layer, p_connection.start_layer))
        {
            return null;
        }
        if (p_connection.target_item instanceof PolylineTrace)
        {
            PolylineTrace to_trace = (PolylineTrace) p_connection.target_item;
            p_board.connect_to_trace(new_instance.first_corner, to_trace, p_ctrl.trace_half_width[p_connection.start_layer], p_ctrl.trace_clearance_class_no);
        }
        if (p_connection.start_item instanceof PolylineTrace)
        {
            PolylineTrace to_trace = (PolylineTrace) p_connection.start_item;
            p_board.connect_to_trace(new_instance.last_corner, to_trace, p_ctrl.trace_half_width[p_connection.target_layer], p_ctrl.trace_clearance_class_no);
        }
        p_board.normalize_traces(p_ctrl.net_no);
        return new_instance;
    }

    /** Creates a new instance of InsertFoundConnectionAlgo */
    private InsertFoundConnectionAlgo(RoutingBoard p_board, AutorouteControl p_ctrl)
    {
        this.board = p_board;
        this.ctrl = p_ctrl;
    }

    /**
     * Inserts the trace by shoving aside obstacle traces and vias.
     * Returns false, that was not possible for the whole trace.
     */
    private boolean insert_trace(LocateFoundConnectionAlgoAnyAngle.ResultItem p_trace)
    {
        if (p_trace.corners.length == 1)
        {
            this.last_corner = p_trace.corners[0];
            return true;
        }
        boolean result = true;

        // switch off correcting connection to pin because it may get wrong in inserting the polygon line for line.
        double saved_edge_to_turn_dist = board.rules.get_pin_edge_to_turn_dist();
        board.rules.set_pin_edge_to_turn_dist(-1);

        // Look for pins att the start and the end of p_trace in case that neckdown is necessecary.
        eu.mihosoft.freerouting.board.Pin start_pin = null;
        eu.mihosoft.freerouting.board.Pin end_pin = null;
        if (ctrl.with_neckdown)
        {
            ItemSelectionFilter item_filter = new ItemSelectionFilter(ItemSelectionFilter.SelectableChoices.PINS);
            Point curr_end_corner = p_trace.corners[0];
            for (int i = 0; i < 2; ++i)
            {
                Set picked_items = this.board.pick_items(curr_end_corner, p_trace.layer, item_filter);
                for (Item curr_item : picked_items)
                {
                    eu.mihosoft.freerouting.board.Pin curr_pin = (eu.mihosoft.freerouting.board.Pin) curr_item;
                    if (curr_pin.contains_net(ctrl.net_no) && curr_pin.get_center().equals(curr_end_corner))
                    {
                        if (i == 0)
                        {
                            start_pin = curr_pin;
                        }
                        else
                        {
                            end_pin = curr_pin;
                        }
                    }
                }
                curr_end_corner = p_trace.corners[p_trace.corners.length - 1];
            }
        }
        int[] net_no_arr = new int[1];
        net_no_arr[0] = ctrl.net_no;

        int from_corner_no = 0;
        for (int i = 1; i < p_trace.corners.length; ++i)
        {
            Point[] curr_corner_arr = new Point[i - from_corner_no + 1];
            for (int j = from_corner_no; j <= i; ++j)
            {
                curr_corner_arr[j - from_corner_no] = p_trace.corners[j];
            }
            Polyline insert_polyline = new Polyline(curr_corner_arr);
            Point ok_point = board.insert_forced_trace_polyline(insert_polyline,
                    ctrl.trace_half_width[p_trace.layer], p_trace.layer, net_no_arr, ctrl.trace_clearance_class_no,
                    ctrl.max_shove_trace_recursion_depth, ctrl.max_shove_via_recursion_depth,
                    ctrl.max_spring_over_recursion_depth, Integer.MAX_VALUE, ctrl.pull_tight_accuracy, true, null);
            boolean neckdown_inserted = false;
            if (ok_point != null && ok_point != insert_polyline.last_corner() && ctrl.with_neckdown && curr_corner_arr.length == 2)
            {
                neckdown_inserted = insert_neckdown(ok_point, curr_corner_arr[1], p_trace.layer, start_pin, end_pin);
            }
            if (ok_point == insert_polyline.last_corner() || neckdown_inserted)
            {
                from_corner_no = i;
            }
            else if (ok_point == insert_polyline.first_corner() && i != p_trace.corners.length - 1)
            {
                // if ok_point == insert_polyline.first_corner() the spring over may have failed.
                // Spring over may correct the situation because an insertion, which is ok with clearance compensation
                // may cause violations without clearance compensation.
                // In this case repeating the insertion with more distant corners may allow the spring_over to correct the situation.
                if (from_corner_no > 0)
                {
                    // p_trace.corners[i] may be inside the offset for the substitute trace around
                    // a spring_over obstacle (if clearance compensation is off).
                    if (curr_corner_arr.length < 3)
                    {
                        // first correction
                        --from_corner_no;
                    }
                }
                if (board.get_test_level().ordinal() >= TestLevel.ALL_DEBUGGING_OUTPUT.ordinal())
                {
                    System.out.println("InsertFoundConnectionAlgo: violation corrected");
                }
            }
            else
            {
                result = false;
                break;
            }
        }
        if (board.get_test_level().ordinal() < TestLevel.ALL_DEBUGGING_OUTPUT.ordinal())
        {
            for (int i = 0; i < p_trace.corners.length - 1; ++i)
            {
                Trace trace_stub = board.get_trace_tail(p_trace.corners[i], p_trace.layer, net_no_arr);
                if (trace_stub != null)
                {
                    board.remove_item(trace_stub);
                }
            }
        }
        board.rules.set_pin_edge_to_turn_dist(saved_edge_to_turn_dist);
        if (this.first_corner == null)
        {
            this.first_corner = p_trace.corners[0];
        }
        this.last_corner = p_trace.corners[p_trace.corners.length - 1];
        return result;
    }

    boolean insert_neckdown(Point p_from_corner, Point p_to_corner, int p_layer, eu.mihosoft.freerouting.board.Pin p_start_pin, eu.mihosoft.freerouting.board.Pin p_end_pin)
    {
        if (p_start_pin != null)
        {
            Point ok_point = try_neck_down(p_to_corner, p_from_corner, p_layer, p_start_pin, true);
            if (ok_point == p_from_corner)
            {
                return true;
            }
        }
        if (p_end_pin != null)
        {
            Point ok_point = try_neck_down(p_from_corner, p_to_corner, p_layer, p_end_pin, false);
            if (ok_point == p_to_corner)
            {
                return true;
            }
        }
        return false;
    }

    private Point try_neck_down(Point p_from_corner, Point p_to_corner, int p_layer, eu.mihosoft.freerouting.board.Pin p_pin, boolean p_at_start)
    {
        if (!p_pin.is_on_layer(p_layer))
        {
            return null;
        }
        FloatPoint pin_center = p_pin.get_center().to_float();
        double curr_clearance =
                this.board.rules.clearance_matrix.value(ctrl.trace_clearance_class_no, p_pin.clearance_class_no(), p_layer);
        double pin_neck_down_distance =
                2 * (0.5 * p_pin.get_max_width(p_layer) + curr_clearance);
        if (pin_center.distance(p_to_corner.to_float()) >= pin_neck_down_distance)
        {
            return null;
        }

        int neck_down_halfwidth = p_pin.get_trace_neckdown_halfwidth(p_layer);
        if (neck_down_halfwidth >= ctrl.trace_half_width[p_layer])
        {
            return null;
        }

        FloatPoint float_from_corner = p_from_corner.to_float();
        FloatPoint float_to_corner = p_to_corner.to_float();

        final int TOLERANCE = 2;

        int[] net_no_arr = new int[1];
        net_no_arr[0] = ctrl.net_no;

        double ok_length = board.check_trace_segment(p_from_corner, p_to_corner, p_layer, net_no_arr,
                ctrl.trace_half_width[p_layer], ctrl.trace_clearance_class_no, true);
        if (ok_length >= Integer.MAX_VALUE)
        {
            return p_from_corner;
        }
        ok_length -= TOLERANCE;
        Point neck_down_end_point;
        if (ok_length <= TOLERANCE)
        {
            neck_down_end_point = p_from_corner;
        }
        else
        {
            FloatPoint float_neck_down_end_point = float_from_corner.change_length(float_to_corner, ok_length);
            neck_down_end_point = float_neck_down_end_point.round();
            // add a corner in case  neck_down_end_point is not exactly on the line from p_from_corner to p_to_corner
            boolean horizontal_first =
                    Math.abs(float_from_corner.x - float_neck_down_end_point.x) >=
                    Math.abs(float_from_corner.y - float_neck_down_end_point.y);
            IntPoint add_corner =
                    LocateFoundConnectionAlgo.calculate_additional_corner(float_from_corner, float_neck_down_end_point,
                    horizontal_first, board.rules.get_trace_angle_restriction()).round();
            Point curr_ok_point = board.insert_forced_trace_segment(p_from_corner,
                    add_corner, ctrl.trace_half_width[p_layer], p_layer, net_no_arr, ctrl.trace_clearance_class_no,
                    ctrl.max_shove_trace_recursion_depth,
                    ctrl.max_shove_via_recursion_depth, ctrl.max_spring_over_recursion_depth, Integer.MAX_VALUE,
                    ctrl.pull_tight_accuracy, true, null);
            if (curr_ok_point != add_corner)
            {
                return p_from_corner;
            }
            curr_ok_point = board.insert_forced_trace_segment(add_corner,
                    neck_down_end_point, ctrl.trace_half_width[p_layer], p_layer, net_no_arr, ctrl.trace_clearance_class_no,
                    ctrl.max_shove_trace_recursion_depth,
                    ctrl.max_shove_via_recursion_depth, ctrl.max_spring_over_recursion_depth, Integer.MAX_VALUE,
                    ctrl.pull_tight_accuracy, true, null);
            if (curr_ok_point != neck_down_end_point)
            {
                return p_from_corner;
            }
            add_corner =
                    LocateFoundConnectionAlgo.calculate_additional_corner(float_neck_down_end_point, float_to_corner,
                    !horizontal_first, board.rules.get_trace_angle_restriction()).round();
            if (!add_corner.equals(p_to_corner))
            {
                curr_ok_point = board.insert_forced_trace_segment(neck_down_end_point, add_corner,
                        ctrl.trace_half_width[p_layer], p_layer, net_no_arr, ctrl.trace_clearance_class_no,
                        ctrl.max_shove_trace_recursion_depth,
                        ctrl.max_shove_via_recursion_depth, ctrl.max_spring_over_recursion_depth, Integer.MAX_VALUE,
                        ctrl.pull_tight_accuracy, true, null);
                if (curr_ok_point != add_corner)
                {
                    return p_from_corner;
                }
                neck_down_end_point = add_corner;
            }
        }

        Point ok_point = board.insert_forced_trace_segment(neck_down_end_point,
                p_to_corner, neck_down_halfwidth, p_layer, net_no_arr, ctrl.trace_clearance_class_no,
                ctrl.max_shove_trace_recursion_depth,
                ctrl.max_shove_via_recursion_depth, ctrl.max_spring_over_recursion_depth, Integer.MAX_VALUE,
                ctrl.pull_tight_accuracy, true, null);
        return ok_point;
    }

    /**
     * Searchs the cheapest via masks containing p_from_layer and p_to_layer, so that a forced via
     * is possible at p_location with this mask and inserts the via.
     * Returns false, if no suitable via mmask was found or if the algorithm failed.
     */
    private boolean insert_via(Point p_location, int p_from_layer, int p_to_layer)
    {
        if (p_from_layer == p_to_layer)
        {
            return true; // no via necessary
        }
        int from_layer;
        int to_layer;
        // sort the input layers
        if (p_from_layer < p_to_layer)
        {
            from_layer = p_from_layer;
            to_layer = p_to_layer;
        }
        else
        {
            from_layer = p_to_layer;
            to_layer = p_from_layer;
        }
        int[] net_no_arr = new int[1];
        net_no_arr[0] = ctrl.net_no;
        ViaInfo via_info = null;
        for (int i = 0; i < this.ctrl.via_rule.via_count(); ++i)
        {
            ViaInfo curr_via_info = this.ctrl.via_rule.get_via(i);
            Padstack curr_via_padstack = curr_via_info.get_padstack();
            if (curr_via_padstack.from_layer() > from_layer || curr_via_padstack.to_layer() < to_layer)
            {
                continue;
            }
            if (ForcedViaAlgo.check(curr_via_info, p_location, net_no_arr,
                    this.ctrl.max_shove_trace_recursion_depth, this.ctrl.max_shove_via_recursion_depth, this.board))
            {
                via_info = curr_via_info;
                break;
            }
        }
        if (via_info == null)
        {
            if (this.board.get_test_level().ordinal() >= TestLevel.CRITICAL_DEBUGGING_OUTPUT.ordinal())
            {
                System.out.print("InsertFoundConnectionAlgo: via mask not found for net ");
                System.out.println(ctrl.net_no);
            }
            return false;
        }
        // insert the via
        if (!ForcedViaAlgo.insert(via_info, p_location, net_no_arr,
                this.ctrl.trace_clearance_class_no, this.ctrl.trace_half_width,
                this.ctrl.max_shove_trace_recursion_depth, this.ctrl.max_shove_via_recursion_depth, this.board))
        {
            if (this.board.get_test_level().ordinal() >= TestLevel.CRITICAL_DEBUGGING_OUTPUT.ordinal())
            {
                System.out.print("InsertFoundConnectionAlgo: forced via failed for net ");
                System.out.println(ctrl.net_no);
            }
            return false;
        }
        return true;
    }
    private final RoutingBoard board;
    private final AutorouteControl ctrl;
    private IntPoint last_corner = null;
    private IntPoint first_corner = null;
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy