Skip to content

Commit

Permalink
makedep: Module dependency in nested includes
Browse files Browse the repository at this point in the history
Nested includes are tracked for the purpose of include flags (-I), but
not with respect to the content within those files.  This patch tracks
the module usage statements (`use ...`) inside of any include files and
adds them to the Makefile rules of the top-level file.

This was implemented within the `nested_inc` function by adding a new
argument.
  • Loading branch information
marshallward authored and adcroft committed Oct 5, 2023
1 parent 2047676 commit 13a1e7e
Showing 1 changed file with 25 additions and 10 deletions.
35 changes: 25 additions & 10 deletions ac/makedep
Original file line number Diff line number Diff line change
Expand Up @@ -150,7 +150,10 @@ def create_deps(src_dirs, makefile, debug, exec_target, fc_rule,
dep for pair in zip(found_mods, found_objs) for dep in pair
]
missing_mods = [m for m in o2uses[o] if m not in all_modules]
incs = nested_inc(o2h[o] + o2inc[o], f2F)

incs, inc_used = nested_inc(o2h[o] + o2inc[o], f2F)
inc_mods = [u for u in inc_used if u not in found_mods and u in all_modules]

incdeps = sorted(set([f2F[f] for f in incs if f in f2F]))
incargs = sorted(set(['-I'+os.path.dirname(f) for f in incdeps]))
if debug:
Expand All @@ -167,7 +170,7 @@ def create_deps(src_dirs, makefile, debug, exec_target, fc_rule,
print("# program:", ' '.join(o2prg[o]), file=file)
if o2mods[o]:
print(' '.join(o2mods[o])+':', o, file=file)
print(o + ':', o2F90[o], ' '.join(incdeps+found_deps), file=file)
print(o + ':', o2F90[o], ' '.join(inc_mods + incdeps + found_deps), file=file)
print('\t'+fc_rule, ' '.join(incargs), file=file)

# Write rule for each object from C
Expand Down Expand Up @@ -243,21 +246,30 @@ def link_obj(obj, o2uses, mod2o, all_modules):
def nested_inc(inc_files, f2F):
"""List of all files included by "inc_files", either by #include or F90
include."""
hlst = []
used_mods = set()

def recur(hfile):
if hfile not in f2F.keys():
return
_, _, cpp, inc, _, _ = scan_fortran_file(f2F[hfile])

_, used, cpp, inc, _, _ = scan_fortran_file(f2F[hfile])

# Record any module updates inside of include files
used_mods.update(used)

if len(cpp) + len(inc) > 0:
for h in cpp+inc:
if h not in hlst and h in f2F.keys():
recur(h)
hlst.append(h)
return
return
hlst = []

for h in inc_files:
recur(h)
return inc_files + sorted(set(hlst))

return inc_files + sorted(set(hlst)), used_mods


def scan_fortran_file(src_file):
Expand All @@ -268,8 +280,10 @@ def scan_fortran_file(src_file):
lines = file.readlines()

external_namespace = True
# True if we are in the external (i.e. global) namespace

file_has_externals = False
# True if the file contains any external objects

for line in lines:
match = re_module.match(line.lower())
Expand Down Expand Up @@ -321,17 +335,18 @@ def object_file(src_file):
def find_files(src_dirs):
"""Return sorted list of all source files starting from each directory in
the list "src_dirs"."""

# TODO: Make this a user-defined argument
extensions = ('.f90', '.f', '.c', '.inc', '.h', '.fh')

files = []

for path in src_dirs:
if not os.path.isdir(path):
raise ValueError("Directory '{}' was not found".format(path))
for p, d, f in os.walk(os.path.normpath(path), followlinks=True):
for file in f:
# TODO: use any()
if (file.endswith('.F90') or file.endswith('.f90')
or file.endswith('.f') or file.endswith('.F')
or file.endswith('.h') or file.endswith('.inc')
or file.endswith('.c') or file.endswith('.H')):
if any(file.lower().endswith(ext) for ext in extensions):
files.append(p+'/'+file)
return sorted(set(files))

Expand Down

0 comments on commit 13a1e7e

Please sign in to comment.