29 auto& grid = data.grid();
30 auto& tdata = data.meta();
31 const int num_blades = tdata.num_blades;
32 const int num_pts_blade = tdata.num_pts_blade;
33 const int num_vel_pts_blade = tdata.num_vel_pts_blade;
35 for (
int ib = 0;
ib < num_blades; ++
ib) {
40 const auto start =
ib * num_pts_blade + 1;
41 const auto start_vel =
ib * num_vel_pts_blade;
44 grid.pos, start, num_pts_blade);
46 grid.force, start, num_pts_blade);
48 grid.epsilon, start, num_pts_blade);
50 grid.orientation, start, num_pts_blade);
52 tdata.chord, start, num_pts_blade);
54 tdata.vel_rel, start, num_pts_blade);
56 grid.vel, start_vel, num_vel_pts_blade);
58 grid.vel_pos, start_vel, num_vel_pts_blade);
61 tdata.blades.emplace_back(cv);
63 if (tdata.num_pts_tower > 0) {
64 const int num_pts_tower = tdata.num_pts_tower;
65 const int ntwr_start = num_blades * num_pts_blade + 1;
66 auto& cv = tdata.tower;
74 grid.orientation, ntwr_start, num_pts_tower);
91 auto& tdata = data.meta();
105 const auto& cd = tdata.nacelle_cd;
106 const auto& area = tdata.nacelle_area;
108 std::sqrt(2.0_rt / std::numbers::pi_v<amrex::Real> * cd * area);
110 auto& nac_eps = data.grid().epsilon[0];
111 nac_eps.x() = amrex::max<amrex::Real>(eps, tdata.eps_min.x());
112 nac_eps.y() = amrex::max<amrex::Real>(eps, tdata.eps_min.y());
113 nac_eps.z() = amrex::max<amrex::Real>(eps, tdata.eps_min.z());
116 for (
int ib = 0;
ib < tdata.num_blades; ++
ib) {
117 auto& cv = tdata.blades[
ib];
119 for (
int i = 0; i < tdata.num_pts_blade; ++i) {
120 const auto eps_crd = tdata.eps_chord * cv.chord[i];
122 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
123 cv.epsilon[i][n] = amrex::max<amrex::Real>(
124 tdata.eps_min[n], tdata.eps_inp[n], eps_crd[n]);
129 auto& cv = tdata.tower;
130 for (
int i = 0; i < tdata.num_pts_tower; ++i) {
131 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
132 cv.epsilon[i][n] = amrex::max<amrex::Real>(
133 tdata.eps_min[n], tdata.eps_inp[n], tdata.eps_tower[n]);
142 if (!data.info().is_root_proc) {
146 const auto& cd = data.meta().nacelle_cd;
147 const auto& area = data.meta().nacelle_area;
148 const auto& cd_area = cd * area;
149 const auto& ext_tdata = data.meta().ext_data;
150 const auto& rho = data.meta().density;
152 const auto& eps = data.grid().epsilon[0].x();
155 ext_tdata.fluid_velocity(0)[0], ext_tdata.fluid_velocity(1)[0],
156 ext_tdata.fluid_velocity(2)[0]};
157 amrex::Real correction = 0.0_rt;
162 correction = 1.0_rt / fac;
165 0.5_rt * rho * cd_area *
vs::mag(vel) * correction * correction;
167 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
168 ext_tdata.force(dir)[0] =
static_cast<float>(coeff * vel[dir]);
198 if (!data.info().actuator_in_proc) {
206 const auto dsize = data.grid().pos.size() * 15;
207 amrex::Vector<float> buf(dsize);
211 if (data.info().is_root_proc) {
212 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter1");
213 const auto& ext_tdata = data.meta().ext_data;
214 auto it = buf.begin();
215 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
217 ext_tdata.force(dir),
218 ext_tdata.force(dir) + ext_tdata.length_force(dir), it);
219 std::advance(it, ext_tdata.length_force(dir));
221 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
223 ext_tdata.position_at_force(dir),
224 ext_tdata.position_at_force(dir) +
225 ext_tdata.length_position_at_force(dir),
227 std::advance(it, ext_tdata.length_position_at_force(dir));
231 std::copy(ext_tdata.orientation(),
232 ext_tdata.orientation() + ext_tdata.length_orientation(), it);
237 const auto& procs = data.info().procs;
238 const int tag = 1001;
239 if (data.info().is_root_proc) {
240 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
241 for (
const int ip : procs) {
242 if (ip == data.info().root_proc) {
246 amrex::ParallelDescriptor::Send(
247 buf.data(), dsize, ip, tag, data.meta().tcomm);
250 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter2");
251 amrex::ParallelDescriptor::Recv(
252 buf.data(), dsize, data.info().root_proc, tag, data.meta().tcomm);
258 BL_PROFILE(
"amr-wind::actuator::external::compute_force_op::scatter3");
259 const auto& bp = data.info().base_pos;
260 auto& grid = data.grid();
261 const auto& npts = grid.pos.size();
262 const auto& rho = data.meta().density;
263 const size_t ifx = 0;
264 const size_t ify = ifx + npts;
265 const size_t ifz = ify + npts;
266 const size_t ipx = ifz + npts;
267 const size_t ipy = ipx + npts;
268 const size_t ipz = ipy + npts;
269 const size_t iori = ipz + npts;
271 for (
int i = 0; i < npts; ++i) {
276 grid.force[i].x() = -
static_cast<amrex::Real
>(buf[ifx + i]) / rho;
277 grid.force[i].y() = -
static_cast<amrex::Real
>(buf[ify + i]) / rho;
278 grid.force[i].z() = -
static_cast<amrex::Real
>(buf[ifz + i]) / rho;
282 grid.pos[i].x() =
static_cast<amrex::Real
>(buf[ipx + i]) + bp.x();
283 grid.pos[i].y() =
static_cast<amrex::Real
>(buf[ipy + i]) + bp.y();
284 grid.pos[i].z() =
static_cast<amrex::Real
>(buf[ipz + i]) + bp.z();
293 static_cast<int>(iori) + i * AMREX_SPACEDIM * AMREX_SPACEDIM;
294 for (
int j = 0; j < AMREX_SPACEDIM; ++j) {
295 for (
int k = 0; k < AMREX_SPACEDIM; ++k) {
296 grid.orientation[i][j * AMREX_SPACEDIM + k] =
297 static_cast<amrex::Real
>(
298 buf[off + j + k * AMREX_SPACEDIM]);
304 auto& meta = data.meta();
305 meta.rot_center = grid.pos[0];
308 const auto xvec = grid.orientation[0].x().unit();
310 const auto zvec = xvec ^ yvec;
311 meta.rotor_frame.rows(xvec, yvec.unit(), zvec.unit());