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;
75 grid.orientation, ntwr_start, num_pts_tower);
92 auto& tdata = data.meta();
106 const auto& cd = tdata.nacelle_cd;
107 const auto& area = tdata.nacelle_area;
109 std::sqrt(2.0_rt / std::numbers::pi_v<amrex::Real> * cd * area);
111 auto& nac_eps = data.grid().epsilon[0];
112 nac_eps.x() = amrex::max<amrex::Real>(eps, tdata.eps_min.x());
113 nac_eps.y() = amrex::max<amrex::Real>(eps, tdata.eps_min.y());
114 nac_eps.z() = amrex::max<amrex::Real>(eps, tdata.eps_min.z());
117 for (
int ib = 0;
ib < tdata.num_blades; ++
ib) {
118 auto& cv = tdata.blades[
ib];
120 for (
int i = 0; i < tdata.num_pts_blade; ++i) {
121 const auto eps_crd = tdata.eps_chord * cv.chord[i];
123 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
124 cv.epsilon[i][n] = amrex::max<amrex::Real>(
125 tdata.eps_min[n], tdata.eps_inp[n], eps_crd[n]);
130 auto& cv = tdata.tower;
131 for (
int i = 0; i < tdata.num_pts_tower; ++i) {
132 for (
int n = 0; n < AMREX_SPACEDIM; ++n) {
133 cv.epsilon[i][n] = amrex::max<amrex::Real>(
134 tdata.eps_min[n], tdata.eps_inp[n], tdata.eps_tower[n]);
143 if (!data.info().is_root_proc) {
147 const auto& cd = data.meta().nacelle_cd;
148 const auto& area = data.meta().nacelle_area;
149 const auto& cd_area = cd * area;
150 const auto& ext_tdata = data.meta().ext_data;
151 const auto& rho = data.meta().density;
153 const auto& eps = data.grid().epsilon[0].x();
156 ext_tdata.fluid_velocity(0)[0], ext_tdata.fluid_velocity(1)[0],
157 ext_tdata.fluid_velocity(2)[0]};
158 amrex::Real correction = 0.0_rt;
163 correction = 1.0_rt / fac;
166 0.5_rt * rho * cd_area *
vs::mag(vel) * correction * correction;
168 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
169 ext_tdata.force(dir)[0] =
static_cast<float>(coeff * vel[dir]);
199 if (!data.info().actuator_in_proc) {
207 const auto dsize = data.grid().pos.size() * 15;
208 amrex::Vector<float> buf(dsize);
212 if (data.info().is_root_proc) {
214 "kynema-sgf::actuator::external::compute_force_op::scatter1");
215 const auto& ext_tdata = data.meta().ext_data;
216 auto it = buf.begin();
217 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
219 ext_tdata.force(dir),
220 ext_tdata.force(dir) + ext_tdata.length_force(dir), it);
221 std::advance(it, ext_tdata.length_force(dir));
223 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
225 ext_tdata.position_at_force(dir),
226 ext_tdata.position_at_force(dir) +
227 ext_tdata.length_position_at_force(dir),
229 std::advance(it, ext_tdata.length_position_at_force(dir));
233 std::copy(ext_tdata.orientation(),
234 ext_tdata.orientation() + ext_tdata.length_orientation(), it);
239 const auto& procs = data.info().procs;
240 const int tag = 1001;
241 if (data.info().is_root_proc) {
243 "kynema-sgf::actuator::external::compute_force_op::scatter2");
244 for (
const int ip : procs) {
245 if (ip == data.info().root_proc) {
249 amrex::ParallelDescriptor::Send(
250 buf.data(), dsize, ip, tag, data.meta().tcomm);
254 "kynema-sgf::actuator::external::compute_force_op::scatter2");
255 amrex::ParallelDescriptor::Recv(
256 buf.data(), dsize, data.info().root_proc, tag, data.meta().tcomm);
263 "kynema-sgf::actuator::external::compute_force_op::scatter3");
264 const auto& bp = data.info().base_pos;
265 auto& grid = data.grid();
266 const auto& npts = grid.pos.size();
267 const auto& rho = data.meta().density;
268 const size_t ifx = 0;
269 const size_t ify = ifx + npts;
270 const size_t ifz = ify + npts;
271 const size_t ipx = ifz + npts;
272 const size_t ipy = ipx + npts;
273 const size_t ipz = ipy + npts;
274 const size_t iori = ipz + npts;
276 for (
int i = 0; i < npts; ++i) {
281 grid.force[i].x() = -
static_cast<amrex::Real
>(buf[ifx + i]) / rho;
282 grid.force[i].y() = -
static_cast<amrex::Real
>(buf[ify + i]) / rho;
283 grid.force[i].z() = -
static_cast<amrex::Real
>(buf[ifz + i]) / rho;
287 grid.pos[i].x() =
static_cast<amrex::Real
>(buf[ipx + i]) + bp.x();
288 grid.pos[i].y() =
static_cast<amrex::Real
>(buf[ipy + i]) + bp.y();
289 grid.pos[i].z() =
static_cast<amrex::Real
>(buf[ipz + i]) + bp.z();
298 static_cast<int>(iori) + i * AMREX_SPACEDIM * AMREX_SPACEDIM;
299 for (
int j = 0; j < AMREX_SPACEDIM; ++j) {
300 for (
int k = 0; k < AMREX_SPACEDIM; ++k) {
301 grid.orientation[i][j * AMREX_SPACEDIM + k] =
302 static_cast<amrex::Real
>(
303 buf[off + j + k * AMREX_SPACEDIM]);
309 auto& meta = data.meta();
310 meta.rot_center = grid.pos[0];
313 const auto xvec = grid.orientation[0].x().unit();
315 const auto zvec = xvec ^ yvec;
316 meta.rotor_frame.rows(xvec, yvec.unit(), zvec.unit());