Skip to content

Commit

Permalink
feat: Support linked lists in functionClass objects with multiple c…
Browse files Browse the repository at this point in the history
…ontained objects
  • Loading branch information
abensonca committed Dec 18, 2024
1 parent 91892f1 commit edc892f
Showing 1 changed file with 123 additions and 76 deletions.
199 changes: 123 additions & 76 deletions perl/Galacticus/Build/SourceTree/Process/FunctionClass.pm
Original file line number Diff line number Diff line change
Expand Up @@ -441,7 +441,7 @@ sub Process_FunctionClass {
$name =~ s/\s//g;
if ( grep {$_ eq lc($name)} @{$potentialNames->{'objects'}} ) {
push(@{$descriptorParameters->{'objects'}},{name => $name, source => $constructorNode->{'directive'}->{'source'}});
} elsif ( exists($nonAbstractClass->{'linkedList'}) && $nonAbstractClass->{'linkedList'}->{'object'} eq $name ) {
} elsif ( exists($nonAbstractClass->{'linkedList'}) && grep {$_ eq $name} split(" ",$nonAbstractClass->{'linkedList'}->{'object'}) ) {
push(@{$descriptorParameters->{'linkedLists'}},$nonAbstractClass->{'linkedList'});
} else {
$supported = -5;
Expand Down Expand Up @@ -2843,14 +2843,17 @@ sub deepCopyLinkedList {
return ("","","",undef())
unless ( exists($class->{'linkedList'}) );
my $linkedList = $class->{'linkedList'};
# Get object names and types.
my @objects = split(" ",$linkedList->{'object' });
my @objectTypes = split(" ",$linkedList->{'objectType'});
# Add variables needed for linked list processing.
push(
@{$linkedListVariables},
{
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item', $linkedList->{'object'}.'destination', $linkedList->{'object'}.'itemNew' ]
variables => [ $linkedList->{'type'}.'item', $linkedList->{'type'}.'destination', $linkedList->{'type'}.'itemNew' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListVariables} );
Expand All @@ -2860,7 +2863,7 @@ sub deepCopyLinkedList {
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item' ]
variables => [ $linkedList->{'type'}.'item' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListResetVariables} );
Expand All @@ -2870,66 +2873,91 @@ sub deepCopyLinkedList {
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item' ]
variables => [ $linkedList->{'type'}.'item' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListFinalizeVariables} );
# Generate code for the walk through the linked list.
$code::variable = $linkedList->{'variable' } ;
$code::object = $linkedList->{'object' } ;
$code::objectType = $linkedList->{'objectType' } ;
$code::objectIntrinsic = exists($linkedList->{'objectIntrinsic'}) ? $linkedList->{'objectIntrinsic'} : "class";
$code::next = $linkedList->{'next' } ;
$code::location = &Galacticus::Build::SourceTree::Process::SourceIntrospection::Location($class->{'node'},$class->{'node'}->{'line'});
$code::debugCode = $debugging ? "if (debugReporting.and.mpiSelf\%isMaster()) call displayMessage(var_str('functionClass[own] (class : ownerName : ownerLoc : objectLoc : sourceLoc): [".$code::objectType."] : ".$code::object." : ')//loc(".$code::object."itemNew)//' : '//loc(".$code::object."itemNew%".$code::object.")//' : '//".&Galacticus::Build::SourceTree::Process::SourceIntrospection::Location($class->{'node'},$class->{'node'}->{'line'},compact => 1).",verbosityLevelSilent)\n" : "";
my $deepCopyCode = fill_in_string(<<'CODE', PACKAGE => 'code');
my $deepCopyCode;
my $deepCopyResetCode;
my $deepCopyFinalizeCode;
for(my $i=0;$i<scalar(@objects);++$i) {
$code::type = $linkedList->{'type' };
$code::variable = $linkedList->{'variable'};
$code::next = $linkedList->{'next' };
$code::object = $objects [$i] ;
$code::objectType = $objectTypes[$i] ;
$code::location = &Galacticus::Build::SourceTree::Process::SourceIntrospection::Location($class->{'node'},$class->{'node'}->{'line'});
$code::debugCode = $debugging ? "if (debugReporting.and.mpiSelf\%isMaster()) call displayMessage(var_str('functionClass[own] (class : ownerName : ownerLoc : objectLoc : sourceLoc): [".$code::objectType."] : ".$code::object." : ')//loc(".$code::object."itemNew)//' : '//loc(".$code::object."itemNew%".$code::object.")//' : '//".&Galacticus::Build::SourceTree::Process::SourceIntrospection::Location($class->{'node'},$class->{'node'}->{'line'},compact => 1).",verbosityLevelSilent)\n" : "";
if ( $i == 0 ) {
$deepCopyCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
destination%{$variable} => null ()
{$object}destination => null ()
{$object}item => self%{$variable}
do while (associated({$object}item))
allocate({$object}itemNew)
if (associated({$object}destination)) then
{$object}destination%{$next} => {$object}itemNew
{$object}destination => {$object}itemNew
CODE
}
$deepCopyCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}destination => null ()
{$type}item => self%{$variable}
do while (associated({$type}item))
CODE
if ( $i == 0 ) {
$deepCopyCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
allocate({$type}itemNew)
if (associated({$type}destination)) then
{$type}destination%{$next} => {$type}itemNew
{$type}destination => {$type}itemNew
else
destination %{$variable} => {$type}itemNew
{$type}destination => {$type}itemNew
end if
CODE
} else {
$deepCopyCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
if (associated({$type}destination)) then
{$type}itemNew => {$type}destination%{$next}
{$type}destination => {$type}destination%{$next}
else
destination %{$variable} => {$object}itemNew
{$object}destination => {$object}itemNew
{$type}itemNew => destination %{$variable}
{$type}destination => destination %{$variable}
end if
nullify({$object}itemNew%{$object})
if (associated({$object}item%{$object})) then
if (associated({$object}item%{$object}%copiedSelf)) then
select type(s => {$object}item%{$object}%copiedSelf)
{$objectIntrinsic} is ({$objectType})
{$object}itemNew%{$object} => s
CODE
}
$deepCopyCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
nullify({$type}itemNew%{$object})
if (associated({$type}item%{$object})) then
if (associated({$type}item%{$object}%copiedSelf)) then
select type(s => {$type}item%{$object}%copiedSelf)
class is ({$objectType})
{$type}itemNew%{$object} => s
class default
call Error_Report('copiedSelf has incorrect type'//{$location})
end select
call {$object}item%{$object}%copiedSelf%referenceCountIncrement()
call {$type}item%{$object}%copiedSelf%referenceCountIncrement()
else
allocate({$object}itemNew%{$object},mold={$object}item%{$object})
call {$object}item%{$object}%deepCopy({$object}itemNew%{$object})
{$object}item%{$object}%copiedSelf => {$object}itemNew%{$object}
call {$object}itemNew%{$object}%autoHook()
allocate({$type}itemNew%{$object},mold={$type}item%{$object})
call {$type}item%{$object}%deepCopy({$type}itemNew%{$object})
{$type}item%{$object}%copiedSelf => {$type}itemNew%{$object}
call {$type}itemNew%{$object}%autoHook()
end if
{$debugCode}
end if
{$object}item => {$object}item%{$next}
{$type}item => {$type}item%{$next}
end do
CODE
my $deepCopyResetCode = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%deepCopyReset()
{$object}item => {$object}item%{$next}
$deepCopyResetCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%deepCopyReset()
{$type}item => {$type}item%{$next}
end do
CODE
my $deepCopyFinalizeCode = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%deepCopyFinalize()
{$object}item => {$object}item%{$next}
$deepCopyFinalizeCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%deepCopyFinalize()
{$type}item => {$type}item%{$next}
end do
CODE
}
my $deepCopyModule = exists($linkedList->{'module'}) ? $linkedList->{'module'} : undef();
return ($deepCopyCode,$deepCopyResetCode,$deepCopyFinalizeCode,$deepCopyModule);
}
Expand All @@ -2942,35 +2970,42 @@ sub stateStoreLinkedList {
return ("","","",undef())
unless ( exists($class->{'linkedList'}) );
my $linkedList = $class->{'linkedList'};
# Get object names.
my @objects = split(" ",$linkedList->{'object'});
# Add variables needed for linked list processing.
push(
@{$linkedListVariables},
{
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item' ]
variables => [ $linkedList->{'type'}.'item' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListVariables} );
# Generate code for the walk through the linked list.
$code::variable = $linkedList->{'variable'};
$code::object = $linkedList->{'object' };
$code::next = $linkedList->{'next' };
my $inputCode = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%stateRestore(stateFile,gslStateFile,stateOperationID)
{$object}item => {$object}item%{$next}
my $inputCode;
my $outputCode;
for(my $i=0;$i<scalar(@objects);++$i) {
$code::type = $linkedList->{'type' };
$code::variable = $linkedList->{'variable'};
$code::next = $linkedList->{'next' };
$code::object = $objects[$i];
$inputCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%stateRestore(stateFile,gslStateFile,stateOperationID)
{$type}item => {$type}item%{$next}
end do
CODE
my $outputCode = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%stateStore(stateFile,gslStateFile,stateOperationID)
{$object}item => {$object}item%{$next}
$outputCode .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%stateStore(stateFile,gslStateFile,stateOperationID)
{$type}item => {$type}item%{$next}
end do
CODE
}
my $deepCopyModule = exists($linkedList->{'module'}) ? $linkedList->{'module'} : undef();
return ($inputCode,$outputCode,$deepCopyModule);
}
Expand All @@ -2983,29 +3018,35 @@ sub allowedParametersLinkedList {
return ("",undef())
unless ( exists($nonAbstractClass->{'linkedList'}) );
my $linkedList = $nonAbstractClass->{'linkedList'};
# Get object names.
my @objects = split(" ",$linkedList->{'object'});
# Add variables needed for linked list processing.
push(
@{$linkedListVariables},
{
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item' ]
variables => [ $linkedList->{'type'}.'item' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListVariables} );
# Generate code for the walk through the linked list.
$code::variable = $linkedList->{'variable'};
$code::object = $linkedList->{'object' };
$code::next = $linkedList->{'next' };
$code::source = $source;
my $iterator = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%allowedParameters(allowedParameters,'{$source}',.true.)
{$object}item => {$object}item%{$next}
my $iterator;
for(my $i=0;$i<scalar(@objects);++$i) {
$code::type = $linkedList->{'type' };
$code::variable = $linkedList->{'variable'};
$code::next = $linkedList->{'next' };
$code::object = $objects[$i];
$code::source = $source;
$iterator .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%allowedParameters(allowedParameters,'{$source}',.true.)
{$type}item => {$type}item%{$next}
end do
CODE
}
my $deepCopyModule = exists($linkedList->{'module'}) ? $linkedList->{'module'} : undef();
return ($iterator,$deepCopyModule);
}
Expand All @@ -3014,28 +3055,34 @@ sub autoDescriptorLinkedList {
# Create auto-descriptor instructions for linked list objects.
my $linkedList = shift();
my $linkedListVariables = shift();
# Get object names.
my @objects = split(" ",$linkedList->{'object'});
# Add variables needed for linked list processing.
push(
@{$linkedListVariables},
{
intrinsic => 'type',
type => $linkedList->{'type'},
attributes => [ 'pointer' ],
variables => [ $linkedList->{'object'}.'item' ]
variables => [ $linkedList->{'type'}.'item' ]
}
)
unless ( grep {$_->{'type'} eq $linkedList->{'type'}} @{$linkedListVariables} );
# Generate code for the walk through the linked list.
$code::variable = $linkedList->{'variable'};
$code::object = $linkedList->{'object' };
$code::next = $linkedList->{'next' };
my $iterator = fill_in_string(<<'CODE', PACKAGE => 'code');
{$object}item => self%{$variable}
do while (associated({$object}item))
call {$object}item%{$object}%descriptor(parameters)
{$object}item => {$object}item%{$next}
my $iterator;
for(my $i=0;$i<scalar(@objects);++$i) {
$code::type = $linkedList->{'type' };
$code::variable = $linkedList->{'variable'};
$code::next = $linkedList->{'next' };
$code::object = $objects[$i];
$iterator .= fill_in_string(<<'CODE', PACKAGE => 'code');
{$type}item => self%{$variable}
do while (associated({$type}item))
call {$type}item%{$object}%descriptor(parameters)
{$type}item => {$type}item%{$next}
end do
CODE
}
my $deepCopyModule = exists($linkedList->{'module'}) ? $linkedList->{'module'} : undef();
return ($iterator,$deepCopyModule);
}
Expand Down

0 comments on commit edc892f

Please sign in to comment.