From 993c0c9102169a01d69bd37e796f14232b3937fe Mon Sep 17 00:00:00 2001 From: "Mads M. Pedersen" <mmpe@dtu.dk> Date: Mon, 14 May 2018 10:25:38 +0200 Subject: [PATCH] implemented openmdao methods for PolygonBoundaryComp and added caching of results --- .../boundary_component.py | 42 +++++++++++++++++-- topfarm/cost_models/fuga/Colonel | 2 +- topfarm/plotting.py | 25 +++++++---- 3 files changed, 57 insertions(+), 12 deletions(-) diff --git a/topfarm/constraint_components/boundary_component.py b/topfarm/constraint_components/boundary_component.py index 7e1292e4..ac977a6c 100644 --- a/topfarm/constraint_components/boundary_component.py +++ b/topfarm/constraint_components/boundary_component.py @@ -190,8 +190,26 @@ class PolygonBoundaryComp(BoundaryComp): self.dEdgeDist_dx = -self.dy / self.length self.dEdgeDist_dy = self.dx / self.length + self._cache_input = None + self._cache_output = None - def calc_distance(self, x, y): + def setup(self): + + # Explicitly size input arrays + self.add_input('turbineX', np.zeros(self.nTurbines), units='m', + desc='x coordinates of turbines in global ref. frame') + self.add_input('turbineY', np.zeros(self.nTurbines), units='m', + desc='y coordinates of turbines in global ref. frame') + + # Explicitly size output array + # (vector with positive elements if turbines outside of hull) + self.add_output('boundaryDistances', np.zeros(self.nTurbines), + desc="signed perpendicular distance from each turbine to each face CCW; + is inside") + + self.declare_partials('boundaryDistances', ['turbineX', 'turbineY']) + #self.declare_partials('boundaryDistances', ['boundaryVertices', 'boundaryNormals'], method='fd') + + def calc_distance_and_gradients(self, x, y): """ distance point(x,y) to edge((x1,y1)->(x2,y2)) +/-: inside/outside @@ -205,7 +223,9 @@ class PolygonBoundaryComp(BoundaryComp): ddist_dx = sign * 2*x-2*x1 / (2 * distance^.5) ddist_dy = sign * 2*y-2*y1 / (2 * distance^.5) """ - + if np.all(np.array([x, y]) == self._cache_input): + return self._cache_output + X, Y = [np.tile(xy, (len(self.x1), 1)).T for xy in [x, y]] # dim = (ntb, nEdges) X1, Y1, X2, Y2, ddist_dX, ddist_dY = [np.tile(xy, (len(x), 1)) for xy in [self.x1, self.y1, self.x2, self.y2, self.dEdgeDist_dx, self.dEdgeDist_dy]] @@ -258,4 +278,20 @@ class PolygonBoundaryComp(BoundaryComp): closest_edge_index = np.argmin(np.abs(distance), 1) - return [np.choose(closest_edge_index, v.T) for v in [distance, ddist_dX, ddist_dY]] + self._cache_input = np.array([x,y]) + self._cache_output = [np.choose(closest_edge_index, v.T) for v in [distance, ddist_dX, ddist_dY]] + return self._cache_output + + def compute(self, inputs, outputs): + turbineX = inputs['turbineX'] + turbineY = inputs['turbineY'] + + outputs['boundaryDistances'] = self.calc_distance_and_gradients(turbineX, turbineY)[0] + + def compute_partials(self, inputs, partials): + turbineX = inputs['turbineX'] + turbineY = inputs['turbineY'] + + _, dx, dy = self.calc_distance_and_gradients(turbineX, turbineY) + partials['boundaryDistances', 'turbineX'] = np.diagflat(dx) + partials['boundaryDistances', 'turbineY'] = np.diagflat(dy) diff --git a/topfarm/cost_models/fuga/Colonel b/topfarm/cost_models/fuga/Colonel index b3d528b5..63d550ac 160000 --- a/topfarm/cost_models/fuga/Colonel +++ b/topfarm/cost_models/fuga/Colonel @@ -1 +1 @@ -Subproject commit b3d528b5bddb5d62f4a6d9f476d7989f5d0f2ad7 +Subproject commit 63d550ac397e2058fc278ecf1488323440a786fc diff --git a/topfarm/plotting.py b/topfarm/plotting.py index a6b67a8f..426b245e 100644 --- a/topfarm/plotting.py +++ b/topfarm/plotting.py @@ -21,9 +21,6 @@ def mypause(interval): class PlotComp(ExplicitComponent): - """ - Evaluates the equation f(x,y) = (x-3)^2 + xy + (y+4)^2 - 3. - """ colors = ['b', 'r', 'm', 'c', 'g', 'y', 'orange', 'indigo', 'grey'] * 100 def __init__(self, memory=10, delay=0.001): @@ -56,11 +53,6 @@ class PlotComp(ExplicitComponent): plt.ylim(ylim) def compute(self, inputs, outputs): - """ - f(x,y) = (x-3)^2 + xy + (y+4)^2 - 3 - - Optimal solution (minimum): x = 6.6667; y = -7.3333 - """ x = inputs['turbineX'] y = inputs['turbineY'] cost = inputs['cost'] @@ -81,3 +73,20 @@ class PlotComp(ExplicitComponent): mypause(self.delay) self.counter += 1 + +class NoPlot(PlotComp): + def __init__(self, *args, **kwargs): + ExplicitComponent.__init__(self) + + def show(self): + pass + + def setup(self): + self.add_input('turbineX', np.zeros(self.n_wt), units='m') + self.add_input('turbineY', np.zeros(self.n_wt), units='m') + self.add_input('cost', 0.) + self.add_input('boundary', np.zeros((self.n_vertices, 2)), units='m') + + + def compute(self, inputs, outputs): + pass \ No newline at end of file -- GitLab