The Perl Toolchain Summit 2025 Needs You: You can help 🙏 Learn more

#!/usr/bin/env perl
use strict;
use Util::H2O::More qw//;
# build and load subroutines
use Inline (
C => 'DATA',
with => qw/OpenMP::Simple/,
);
my $env = OpenMP::Environment->new();
my $aref_orig = [
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
[ 1 .. 10 ],
];
my $expected = [qw/1 2 3 4 5 6 7 8 9 10/];
foreach my $thread_count (qw/1 4 8/) {
$env->omp_num_threads($thread_count);
foreach my $row_orig (@$aref_orig) {
my $aref_new = omp_get_renew_aref($row_orig);
my $seen_elements = shift @$aref_new;
my $seen_threads = shift @$aref_new;
is $seen_elements, scalar @$row_orig, q{PerlOMP_1D_Array_NUM_ELEMENTS works on original ARRAY reference};
is $seen_threads, $thread_count, qq{OMP_NUM_THREADS=$thread_count is respected inside of the, omp parallel section, as expected};
cmp_deeply $aref_new, $expected, qq{Row summed array ref returned as expected from $thread_count OpenMP threads};
cmp_deeply $aref_new, $expected, qq{PerlOMP_1D_Array_TO_1D_INT_ARRAY worked to convert original ARRAY reference to raw C 1D array of floats};
}
}
done_testing;
__DATA__
__C__
/* Custom driver */
AV* omp_get_renew_aref(SV *ARRAY) {
/* boilerplate - updates number of threads to use with what's in $ENV{OMP_NUM_THREADS} */
PerlOMP_UPDATE_WITH_ENV__NUM_THREADS
/* boilerplate - creates an array to return back to perl, named "ret" */
/* note, "ret" can contain anything, when added via "av_push" */
PerlOMP_RET_ARRAY_REF_ret
/* non-boilerplate (for the test, we want this to apply to all rows, though) */
int num_elements = PerlOMP_1D_Array_NUM_ELEMENTS(ARRAY);
av_push(ret, newSViv(num_elements));
/* get 1d array ref into a 1d C array */
int raw_array[num_elements]; // create native 2D array as target
PerlOMP_1D_Array_TO_1D_INT_ARRAY_r(ARRAY, num_elements, raw_array); // call macro to put AoA into native "nodes" array - threaded version
int sum[num_elements];
#pragma omp parallel shared(raw_array,num_elements,sum)
#pragma omp master
av_push(ret, newSViv(omp_get_num_threads()));
#pragma omp for
for(int i=0; i<num_elements; i++) {
sum[i] = raw_array[i];
}
for(int i=0; i<num_elements; i++) {
av_push(ret, newSViv(sum[i]));
}
// AV* 'ret' comes from "PerlOMP_RET_ARRAY_REF_ret" macro called above
return ret;
}