f90mod_rules.py 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. #!/usr/bin/env python
  2. """
  3. Build F90 module support for f2py2e.
  4. Copyright 2000 Pearu Peterson all rights reserved,
  5. Pearu Peterson <pearu@ioc.ee>
  6. Permission to use, modify, and distribute this software is given under the
  7. terms of the NumPy License.
  8. NO WARRANTY IS EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
  9. $Date: 2005/02/03 19:30:23 $
  10. Pearu Peterson
  11. """
  12. from __future__ import division, absolute_import, print_function
  13. __version__ = "$Revision: 1.27 $"[10:-1]
  14. f2py_version = 'See `f2py -v`'
  15. import numpy as np
  16. from . import capi_maps
  17. from . import func2subr
  18. from .crackfortran import undo_rmbadname, undo_rmbadname1
  19. # The eviroment provided by auxfuncs.py is needed for some calls to eval.
  20. # As the needed functions cannot be determined by static inspection of the
  21. # code, it is safest to use import * pending a major refactoring of f2py.
  22. from .auxfuncs import *
  23. options = {}
  24. def findf90modules(m):
  25. if ismodule(m):
  26. return [m]
  27. if not hasbody(m):
  28. return []
  29. ret = []
  30. for b in m['body']:
  31. if ismodule(b):
  32. ret.append(b)
  33. else:
  34. ret = ret + findf90modules(b)
  35. return ret
  36. fgetdims1 = """\
  37. external f2pysetdata
  38. logical ns
  39. integer r,i
  40. integer(%d) s(*)
  41. ns = .FALSE.
  42. if (allocated(d)) then
  43. do i=1,r
  44. if ((size(d,i).ne.s(i)).and.(s(i).ge.0)) then
  45. ns = .TRUE.
  46. end if
  47. end do
  48. if (ns) then
  49. deallocate(d)
  50. end if
  51. end if
  52. if ((.not.allocated(d)).and.(s(1).ge.1)) then""" % np.intp().itemsize
  53. fgetdims2 = """\
  54. end if
  55. if (allocated(d)) then
  56. do i=1,r
  57. s(i) = size(d,i)
  58. end do
  59. end if
  60. flag = 1
  61. call f2pysetdata(d,allocated(d))"""
  62. fgetdims2_sa = """\
  63. end if
  64. if (allocated(d)) then
  65. do i=1,r
  66. s(i) = size(d,i)
  67. end do
  68. !s(r) must be equal to len(d(1))
  69. end if
  70. flag = 2
  71. call f2pysetdata(d,allocated(d))"""
  72. def buildhooks(pymod):
  73. global fgetdims1, fgetdims2
  74. from . import rules
  75. ret = {'f90modhooks': [], 'initf90modhooks': [], 'body': [],
  76. 'need': ['F_FUNC', 'arrayobject.h'],
  77. 'separatorsfor': {'includes0': '\n', 'includes': '\n'},
  78. 'docs': ['"Fortran 90/95 modules:\\n"'],
  79. 'latexdoc': []}
  80. fhooks = ['']
  81. def fadd(line, s=fhooks):
  82. s[0] = '%s\n %s' % (s[0], line)
  83. doc = ['']
  84. def dadd(line, s=doc):
  85. s[0] = '%s\n%s' % (s[0], line)
  86. for m in findf90modules(pymod):
  87. sargs, fargs, efargs, modobjs, notvars, onlyvars = [], [], [], [], [
  88. m['name']], []
  89. sargsp = []
  90. ifargs = []
  91. mfargs = []
  92. if hasbody(m):
  93. for b in m['body']:
  94. notvars.append(b['name'])
  95. for n in m['vars'].keys():
  96. var = m['vars'][n]
  97. if (n not in notvars) and (not l_or(isintent_hide, isprivate)(var)):
  98. onlyvars.append(n)
  99. mfargs.append(n)
  100. outmess('\t\tConstructing F90 module support for "%s"...\n' %
  101. (m['name']))
  102. if onlyvars:
  103. outmess('\t\t Variables: %s\n' % (' '.join(onlyvars)))
  104. chooks = ['']
  105. def cadd(line, s=chooks):
  106. s[0] = '%s\n%s' % (s[0], line)
  107. ihooks = ['']
  108. def iadd(line, s=ihooks):
  109. s[0] = '%s\n%s' % (s[0], line)
  110. vrd = capi_maps.modsign2map(m)
  111. cadd('static FortranDataDef f2py_%s_def[] = {' % (m['name']))
  112. dadd('\\subsection{Fortran 90/95 module \\texttt{%s}}\n' % (m['name']))
  113. if hasnote(m):
  114. note = m['note']
  115. if isinstance(note, list):
  116. note = '\n'.join(note)
  117. dadd(note)
  118. if onlyvars:
  119. dadd('\\begin{description}')
  120. for n in onlyvars:
  121. var = m['vars'][n]
  122. modobjs.append(n)
  123. ct = capi_maps.getctype(var)
  124. at = capi_maps.c2capi_map[ct]
  125. dm = capi_maps.getarrdims(n, var)
  126. dms = dm['dims'].replace('*', '-1').strip()
  127. dms = dms.replace(':', '-1').strip()
  128. if not dms:
  129. dms = '-1'
  130. use_fgetdims2 = fgetdims2
  131. if isstringarray(var):
  132. if 'charselector' in var and 'len' in var['charselector']:
  133. cadd('\t{"%s",%s,{{%s,%s}},%s},'
  134. % (undo_rmbadname1(n), dm['rank'], dms, var['charselector']['len'], at))
  135. use_fgetdims2 = fgetdims2_sa
  136. else:
  137. cadd('\t{"%s",%s,{{%s}},%s},' %
  138. (undo_rmbadname1(n), dm['rank'], dms, at))
  139. else:
  140. cadd('\t{"%s",%s,{{%s}},%s},' %
  141. (undo_rmbadname1(n), dm['rank'], dms, at))
  142. dadd('\\item[]{{}\\verb@%s@{}}' %
  143. (capi_maps.getarrdocsign(n, var)))
  144. if hasnote(var):
  145. note = var['note']
  146. if isinstance(note, list):
  147. note = '\n'.join(note)
  148. dadd('--- %s' % (note))
  149. if isallocatable(var):
  150. fargs.append('f2py_%s_getdims_%s' % (m['name'], n))
  151. efargs.append(fargs[-1])
  152. sargs.append(
  153. 'void (*%s)(int*,int*,void(*)(char*,int*),int*)' % (n))
  154. sargsp.append('void (*)(int*,int*,void(*)(char*,int*),int*)')
  155. iadd('\tf2py_%s_def[i_f2py++].func = %s;' % (m['name'], n))
  156. fadd('subroutine %s(r,s,f2pysetdata,flag)' % (fargs[-1]))
  157. fadd('use %s, only: d => %s\n' %
  158. (m['name'], undo_rmbadname1(n)))
  159. fadd('integer flag\n')
  160. fhooks[0] = fhooks[0] + fgetdims1
  161. dms = eval('range(1,%s+1)' % (dm['rank']))
  162. fadd(' allocate(d(%s))\n' %
  163. (','.join(['s(%s)' % i for i in dms])))
  164. fhooks[0] = fhooks[0] + use_fgetdims2
  165. fadd('end subroutine %s' % (fargs[-1]))
  166. else:
  167. fargs.append(n)
  168. sargs.append('char *%s' % (n))
  169. sargsp.append('char*')
  170. iadd('\tf2py_%s_def[i_f2py++].data = %s;' % (m['name'], n))
  171. if onlyvars:
  172. dadd('\\end{description}')
  173. if hasbody(m):
  174. for b in m['body']:
  175. if not isroutine(b):
  176. print('Skipping', b['block'], b['name'])
  177. continue
  178. modobjs.append('%s()' % (b['name']))
  179. b['modulename'] = m['name']
  180. api, wrap = rules.buildapi(b)
  181. if isfunction(b):
  182. fhooks[0] = fhooks[0] + wrap
  183. fargs.append('f2pywrap_%s_%s' % (m['name'], b['name']))
  184. ifargs.append(func2subr.createfuncwrapper(b, signature=1))
  185. else:
  186. if wrap:
  187. fhooks[0] = fhooks[0] + wrap
  188. fargs.append('f2pywrap_%s_%s' % (m['name'], b['name']))
  189. ifargs.append(
  190. func2subr.createsubrwrapper(b, signature=1))
  191. else:
  192. fargs.append(b['name'])
  193. mfargs.append(fargs[-1])
  194. api['externroutines'] = []
  195. ar = applyrules(api, vrd)
  196. ar['docs'] = []
  197. ar['docshort'] = []
  198. ret = dictappend(ret, ar)
  199. cadd('\t{"%s",-1,{{-1}},0,NULL,(void *)f2py_rout_#modulename#_%s_%s,doc_f2py_rout_#modulename#_%s_%s},' %
  200. (b['name'], m['name'], b['name'], m['name'], b['name']))
  201. sargs.append('char *%s' % (b['name']))
  202. sargsp.append('char *')
  203. iadd('\tf2py_%s_def[i_f2py++].data = %s;' %
  204. (m['name'], b['name']))
  205. cadd('\t{NULL}\n};\n')
  206. iadd('}')
  207. ihooks[0] = 'static void f2py_setup_%s(%s) {\n\tint i_f2py=0;%s' % (
  208. m['name'], ','.join(sargs), ihooks[0])
  209. if '_' in m['name']:
  210. F_FUNC = 'F_FUNC_US'
  211. else:
  212. F_FUNC = 'F_FUNC'
  213. iadd('extern void %s(f2pyinit%s,F2PYINIT%s)(void (*)(%s));'
  214. % (F_FUNC, m['name'], m['name'].upper(), ','.join(sargsp)))
  215. iadd('static void f2py_init_%s(void) {' % (m['name']))
  216. iadd('\t%s(f2pyinit%s,F2PYINIT%s)(f2py_setup_%s);'
  217. % (F_FUNC, m['name'], m['name'].upper(), m['name']))
  218. iadd('}\n')
  219. ret['f90modhooks'] = ret['f90modhooks'] + chooks + ihooks
  220. ret['initf90modhooks'] = ['\tPyDict_SetItemString(d, "%s", PyFortranObject_New(f2py_%s_def,f2py_init_%s));' % (
  221. m['name'], m['name'], m['name'])] + ret['initf90modhooks']
  222. fadd('')
  223. fadd('subroutine f2pyinit%s(f2pysetupfunc)' % (m['name']))
  224. if mfargs:
  225. for a in undo_rmbadname(mfargs):
  226. fadd('use %s, only : %s' % (m['name'], a))
  227. if ifargs:
  228. fadd(' '.join(['interface'] + ifargs))
  229. fadd('end interface')
  230. fadd('external f2pysetupfunc')
  231. if efargs:
  232. for a in undo_rmbadname(efargs):
  233. fadd('external %s' % (a))
  234. fadd('call f2pysetupfunc(%s)' % (','.join(undo_rmbadname(fargs))))
  235. fadd('end subroutine f2pyinit%s\n' % (m['name']))
  236. dadd('\n'.join(ret['latexdoc']).replace(
  237. r'\subsection{', r'\subsubsection{'))
  238. ret['latexdoc'] = []
  239. ret['docs'].append('"\t%s --- %s"' % (m['name'],
  240. ','.join(undo_rmbadname(modobjs))))
  241. ret['routine_defs'] = ''
  242. ret['doc'] = []
  243. ret['docshort'] = []
  244. ret['latexdoc'] = doc[0]
  245. if len(ret['docs']) <= 1:
  246. ret['docs'] = ''
  247. return ret, fhooks[0]