27 auto& grid = data.grid();
28 auto& tdata = data.meta();
29 const int num_blades = tdata.num_blades;
30 const int num_pts_blade = tdata.num_pts_blade;
31 const int num_vel_pts_blade = tdata.num_vel_pts_blade;
33 for (
int ib = 0;
ib < num_blades; ++
ib) {
38 const auto start =
ib * num_pts_blade + 1;
39 const auto start_vel =
ib * num_vel_pts_blade;
42 grid.pos, start, num_pts_blade);
44 grid.force, start, num_pts_blade);
46 grid.epsilon, start, num_pts_blade);
48 grid.orientation, start, num_pts_blade);
50 tdata.chord, start, num_pts_blade);
52 tdata.vel_rel, start, num_pts_blade);
54 grid.vel, start_vel, num_vel_pts_blade);
56 grid.vel_pos, start_vel, num_vel_pts_blade);
59 tdata.blades.emplace_back(cv);
61 if (tdata.num_pts_tower > 0) {
62 const int num_pts_tower = tdata.num_pts_tower;
63 const int ntwr_start = num_blades * num_pts_blade + 1;
64 auto& cv = tdata.tower;
72 grid.orientation, ntwr_start, num_pts_tower);
89 auto& tdata = data.meta();
103 const auto& cd = tdata.nacelle_cd;
104 const auto& area = tdata.nacelle_area;
108 auto& nac_eps = data.grid().epsilon[0];
109 nac_eps.x() = amrex::max<amrex::Real>(eps, tdata.eps_min.x());
110 nac_eps.y() = amrex::max<amrex::Real>(eps, tdata.eps_min.y());
111 nac_eps.z() = amrex::max<amrex::Real>(eps, tdata.eps_min.z());
114 for (
int ib = 0;
ib < tdata.num_blades; ++
ib) {
115 auto& cv = tdata.blades[
ib];
117 for (
int i = 0; i < tdata.num_pts_blade; ++i) {
118 const auto eps_crd = tdata.eps_chord * cv.chord[i];
120 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
121 cv.epsilon[i][n] = amrex::max<amrex::Real>(
122 tdata.eps_min[n], tdata.eps_inp[n], eps_crd[n]);
127 auto& cv = tdata.tower;
128 for (
int i = 0; i < tdata.num_pts_tower; ++i) {
129 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
130 cv.epsilon[i][n] = amrex::max<amrex::Real>(
131 tdata.eps_min[n], tdata.eps_inp[n], tdata.eps_tower[n]);
140 if (!data.info().is_root_proc) {
144 const auto& cd = data.meta().nacelle_cd;
145 const auto& area = data.meta().nacelle_area;
146 const auto& cd_area = cd * area;
147 const auto& ext_tdata = data.meta().ext_data;
148 const auto& rho = data.meta().density;
150 const auto& eps = data.grid().epsilon[0].x();
153 ext_tdata.fluid_velocity(0)[0], ext_tdata.fluid_velocity(1)[0],
154 ext_tdata.fluid_velocity(2)[0]};
155 amrex::Real correction = 0.0_rt;
160 correction = 1.0_rt / fac;
163 0.5_rt * rho * cd_area *
vs::mag(vel) * correction * correction;
165 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
166 ext_tdata.force(dir)[0] =
static_cast<float>(coeff * vel[dir]);
196 if (!data.info().actuator_in_proc) {
204 const auto dsize = data.grid().pos.size() * 15;
205 amrex::Vector<float> buf(dsize);
209 if (data.info().is_root_proc) {
210 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter1");
211 const auto& ext_tdata = data.meta().ext_data;
212 auto it = buf.begin();
213 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
215 ext_tdata.force(dir),
216 ext_tdata.force(dir) + ext_tdata.length_force(dir), it);
217 std::advance(it, ext_tdata.length_force(dir));
219 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
221 ext_tdata.position_at_force(dir),
222 ext_tdata.position_at_force(dir) +
223 ext_tdata.length_position_at_force(dir),
225 std::advance(it, ext_tdata.length_position_at_force(dir));
229 std::copy(ext_tdata.orientation(),
230 ext_tdata.orientation() + ext_tdata.length_orientation(), it);
235 const auto& procs = data.info().procs;
236 const int tag = 1001;
237 if (data.info().is_root_proc) {
238 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
239 for (
const int ip : procs) {
240 if (ip == data.info().root_proc) {
244 amrex::ParallelDescriptor::Send(
245 buf.data(), dsize, ip, tag, data.meta().tcomm);
248 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
249 amrex::ParallelDescriptor::Recv(
250 buf.data(), dsize, data.info().root_proc, tag, data.meta().tcomm);
256 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter3");
257 const auto& bp = data.info().base_pos;
258 auto& grid = data.grid();
259 const auto& npts = grid.pos.size();
260 const auto& rho = data.meta().density;
261 const size_t ifx = 0;
262 const size_t ify = ifx + npts;
263 const size_t ifz = ify + npts;
264 const size_t ipx = ifz + npts;
265 const size_t ipy = ipx + npts;
266 const size_t ipz = ipy + npts;
267 const size_t iori = ipz + npts;
269 for (
int i = 0; i < npts; ++i) {
274 grid.force[i].x() = -
static_cast<amrex::Real
>(buf[ifx + i]) / rho;
275 grid.force[i].y() = -
static_cast<amrex::Real
>(buf[ify + i]) / rho;
276 grid.force[i].z() = -
static_cast<amrex::Real
>(buf[ifz + i]) / rho;
280 grid.pos[i].x() =
static_cast<amrex::Real
>(buf[ipx + i]) + bp.x();
281 grid.pos[i].y() =
static_cast<amrex::Real
>(buf[ipy + i]) + bp.y();
282 grid.pos[i].z() =
static_cast<amrex::Real
>(buf[ipz + i]) + bp.z();
291 static_cast<int>(iori) + i * AMREX_SPACEDIM * AMREX_SPACEDIM;
292 for (
int j = 0; j < AMREX_SPACEDIM; ++j) {
293 for (
int k = 0; k < AMREX_SPACEDIM; ++k) {
294 grid.orientation[i][j * AMREX_SPACEDIM + k] =
295 static_cast<amrex::Real
>(
296 buf[off + j + k * AMREX_SPACEDIM]);
302 auto& meta = data.meta();
303 meta.rot_center = grid.pos[0];
306 const auto xvec = grid.orientation[0].x().unit();
308 const auto zvec = xvec ^ yvec;
309 meta.rotor_frame.rows(xvec, yvec.unit(), zvec.unit());