28 auto& grid = data.grid();
29 auto& tdata = data.meta();
30 const int num_blades = tdata.num_blades;
31 const int num_pts_blade = tdata.num_pts_blade;
32 const int num_vel_pts_blade = tdata.num_vel_pts_blade;
34 for (
int ib = 0;
ib < num_blades; ++
ib) {
39 const auto start =
ib * num_pts_blade + 1;
40 const auto start_vel =
ib * num_vel_pts_blade;
43 grid.pos, start, num_pts_blade);
45 grid.force, start, num_pts_blade);
47 grid.epsilon, start, num_pts_blade);
49 grid.orientation, start, num_pts_blade);
51 tdata.chord, start, num_pts_blade);
53 tdata.vel_rel, start, num_pts_blade);
55 grid.vel, start_vel, num_vel_pts_blade);
57 grid.vel_pos, start_vel, num_vel_pts_blade);
60 tdata.blades.emplace_back(cv);
62 if (tdata.num_pts_tower > 0) {
63 const int num_pts_tower = tdata.num_pts_tower;
64 const int ntwr_start = num_blades * num_pts_blade + 1;
65 auto& cv = tdata.tower;
73 grid.orientation, ntwr_start, num_pts_tower);
90 auto& tdata = data.meta();
104 const auto& cd = tdata.nacelle_cd;
105 const auto& area = tdata.nacelle_area;
107 std::sqrt(2.0_rt / std::numbers::pi_v<amrex::Real> * cd * area);
109 auto& nac_eps = data.grid().epsilon[0];
110 nac_eps.x() = amrex::max<amrex::Real>(eps, tdata.eps_min.x());
111 nac_eps.y() = amrex::max<amrex::Real>(eps, tdata.eps_min.y());
112 nac_eps.z() = amrex::max<amrex::Real>(eps, tdata.eps_min.z());
115 for (
int ib = 0;
ib < tdata.num_blades; ++
ib) {
116 auto& cv = tdata.blades[
ib];
118 for (
int i = 0; i < tdata.num_pts_blade; ++i) {
119 const auto eps_crd = tdata.eps_chord * cv.chord[i];
121 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
122 cv.epsilon[i][n] = amrex::max<amrex::Real>(
123 tdata.eps_min[n], tdata.eps_inp[n], eps_crd[n]);
128 auto& cv = tdata.tower;
129 for (
int i = 0; i < tdata.num_pts_tower; ++i) {
130 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
131 cv.epsilon[i][n] = amrex::max<amrex::Real>(
132 tdata.eps_min[n], tdata.eps_inp[n], tdata.eps_tower[n]);
141 if (!data.info().is_root_proc) {
145 const auto& cd = data.meta().nacelle_cd;
146 const auto& area = data.meta().nacelle_area;
147 const auto& cd_area = cd * area;
148 const auto& ext_tdata = data.meta().ext_data;
149 const auto& rho = data.meta().density;
151 const auto& eps = data.grid().epsilon[0].x();
154 ext_tdata.fluid_velocity(0)[0], ext_tdata.fluid_velocity(1)[0],
155 ext_tdata.fluid_velocity(2)[0]};
156 amrex::Real correction = 0.0_rt;
161 correction = 1.0_rt / fac;
164 0.5_rt * rho * cd_area *
vs::mag(vel) * correction * correction;
166 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
167 ext_tdata.force(dir)[0] =
static_cast<float>(coeff * vel[dir]);
197 if (!data.info().actuator_in_proc) {
205 const auto dsize = data.grid().pos.size() * 15;
206 amrex::Vector<float> buf(dsize);
210 if (data.info().is_root_proc) {
211 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter1");
212 const auto& ext_tdata = data.meta().ext_data;
213 auto it = buf.begin();
214 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
216 ext_tdata.force(dir),
217 ext_tdata.force(dir) + ext_tdata.length_force(dir), it);
218 std::advance(it, ext_tdata.length_force(dir));
220 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
222 ext_tdata.position_at_force(dir),
223 ext_tdata.position_at_force(dir) +
224 ext_tdata.length_position_at_force(dir),
226 std::advance(it, ext_tdata.length_position_at_force(dir));
230 std::copy(ext_tdata.orientation(),
231 ext_tdata.orientation() + ext_tdata.length_orientation(), it);
236 const auto& procs = data.info().procs;
237 const int tag = 1001;
238 if (data.info().is_root_proc) {
239 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
240 for (
const int ip : procs) {
241 if (ip == data.info().root_proc) {
245 amrex::ParallelDescriptor::Send(
246 buf.data(), dsize, ip, tag, data.meta().tcomm);
249 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
250 amrex::ParallelDescriptor::Recv(
251 buf.data(), dsize, data.info().root_proc, tag, data.meta().tcomm);
257 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter3");
258 const auto& bp = data.info().base_pos;
259 auto& grid = data.grid();
260 const auto& npts = grid.pos.size();
261 const auto& rho = data.meta().density;
262 const size_t ifx = 0;
263 const size_t ify = ifx + npts;
264 const size_t ifz = ify + npts;
265 const size_t ipx = ifz + npts;
266 const size_t ipy = ipx + npts;
267 const size_t ipz = ipy + npts;
268 const size_t iori = ipz + npts;
270 for (
int i = 0; i < npts; ++i) {
275 grid.force[i].x() = -
static_cast<amrex::Real
>(buf[ifx + i]) / rho;
276 grid.force[i].y() = -
static_cast<amrex::Real
>(buf[ify + i]) / rho;
277 grid.force[i].z() = -
static_cast<amrex::Real
>(buf[ifz + i]) / rho;
281 grid.pos[i].x() =
static_cast<amrex::Real
>(buf[ipx + i]) + bp.x();
282 grid.pos[i].y() =
static_cast<amrex::Real
>(buf[ipy + i]) + bp.y();
283 grid.pos[i].z() =
static_cast<amrex::Real
>(buf[ipz + i]) + bp.z();
292 static_cast<int>(iori) + i * AMREX_SPACEDIM * AMREX_SPACEDIM;
293 for (
int j = 0; j < AMREX_SPACEDIM; ++j) {
294 for (
int k = 0; k < AMREX_SPACEDIM; ++k) {
295 grid.orientation[i][j * AMREX_SPACEDIM + k] =
296 static_cast<amrex::Real
>(
297 buf[off + j + k * AMREX_SPACEDIM]);
303 auto& meta = data.meta();
304 meta.rot_center = grid.pos[0];
307 const auto xvec = grid.orientation[0].x().unit();
309 const auto zvec = xvec ^ yvec;
310 meta.rotor_frame.rows(xvec, yvec.unit(), zvec.unit());