| | import re |
| | import itertools |
| |
|
| | def isBlank(line): |
| | return not line |
| | def isLabel(line): |
| | return line[0].isdigit() |
| | def isComment(line): |
| | return line[0] != ' ' |
| | def isContinuation(line): |
| | return line[5] != ' ' |
| |
|
| | COMMENT, STATEMENT, CONTINUATION = 0, 1, 2 |
| | def lineType(line): |
| | """Return the type of a line of Fortan code.""" |
| | if isBlank(line): |
| | return COMMENT |
| | elif isLabel(line): |
| | return STATEMENT |
| | elif isComment(line): |
| | return COMMENT |
| | elif isContinuation(line): |
| | return CONTINUATION |
| | else: |
| | return STATEMENT |
| |
|
| | class LineIterator(object): |
| | """LineIterator(iterable) |
| | |
| | Return rstrip()'d lines from iterable, while keeping a count of the |
| | line number in the .lineno attribute. |
| | """ |
| | def __init__(self, iterable): |
| | object.__init__(self) |
| | self.iterable = iter(iterable) |
| | self.lineno = 0 |
| | def __iter__(self): |
| | return self |
| | def next(self): |
| | self.lineno += 1 |
| | line = self.iterable.next() |
| | line = line.rstrip() |
| | return line |
| |
|
| | class PushbackIterator(object): |
| | """PushbackIterator(iterable) |
| | |
| | Return an iterator for which items can be pushed back into. |
| | Call the .pushback(item) method to have item returned as the next |
| | value of .next(). |
| | """ |
| | def __init__(self, iterable): |
| | object.__init__(self) |
| | self.iterable = iter(iterable) |
| | self.buffer = [] |
| |
|
| | def __iter__(self): |
| | return self |
| |
|
| | def next(self): |
| | if self.buffer: |
| | return self.buffer.pop() |
| | else: |
| | return self.iterable.next() |
| |
|
| | def pushback(self, item): |
| | self.buffer.append(item) |
| |
|
| | def fortranSourceLines(fo): |
| | """Return an iterator over statement lines of a Fortran source file. |
| | |
| | Comment and blank lines are stripped out, and continuation lines are |
| | merged. |
| | """ |
| | numberingiter = LineIterator(fo) |
| | |
| | with_extra = itertools.chain(numberingiter, ['']) |
| | pushbackiter = PushbackIterator(with_extra) |
| | for line in pushbackiter: |
| | t = lineType(line) |
| | if t == COMMENT: |
| | continue |
| | elif t == STATEMENT: |
| | lines = [line] |
| | |
| | |
| | for next_line in pushbackiter: |
| | t = lineType(next_line) |
| | if t == CONTINUATION: |
| | lines.append(next_line[6:]) |
| | else: |
| | pushbackiter.pushback(next_line) |
| | break |
| | yield numberingiter.lineno, ''.join(lines) |
| | else: |
| | raise ValueError("jammed: continuation line not expected: %s:%d" % |
| | (fo.name, numberingiter.lineno)) |
| |
|
| | def getDependencies(filename): |
| | """For a Fortran source file, return a list of routines declared as EXTERNAL |
| | in it. |
| | """ |
| | fo = open(filename) |
| | external_pat = re.compile(r'^\s*EXTERNAL\s', re.I) |
| | routines = [] |
| | for lineno, line in fortranSourceLines(fo): |
| | m = external_pat.match(line) |
| | if m: |
| | names = line = line[m.end():].strip().split(',') |
| | names = [n.strip().lower() for n in names] |
| | names = [n for n in names if n] |
| | routines.extend(names) |
| | fo.close() |
| | return routines |
| |
|