func2subr.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. """
  2. Rules for building C/API module with f2py2e.
  3. Copyright 1999 -- 2011 Pearu Peterson all rights reserved.
  4. Copyright 2011 -- present NumPy Developers.
  5. Permission to use, modify, and distribute this software is given under the
  6. terms of the NumPy License.
  7. NO WARRANTY IS EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
  8. """
  9. import copy
  10. from ._isocbind import isoc_kindmap
  11. from .auxfuncs import (
  12. getfortranname,
  13. isexternal,
  14. isfunction,
  15. isfunction_wrap,
  16. isintent_in,
  17. isintent_out,
  18. islogicalfunction,
  19. ismoduleroutine,
  20. isscalar,
  21. issubroutine,
  22. issubroutine_wrap,
  23. outmess,
  24. show,
  25. )
  26. def var2fixfortran(vars, a, fa=None, f90mode=None):
  27. if fa is None:
  28. fa = a
  29. if a not in vars:
  30. show(vars)
  31. outmess(f'var2fixfortran: No definition for argument "{a}".\n')
  32. return ''
  33. if 'typespec' not in vars[a]:
  34. show(vars[a])
  35. outmess(f'var2fixfortran: No typespec for argument "{a}".\n')
  36. return ''
  37. vardef = vars[a]['typespec']
  38. if vardef == 'type' and 'typename' in vars[a]:
  39. vardef = f"{vardef}({vars[a]['typename']})"
  40. selector = {}
  41. lk = ''
  42. if 'kindselector' in vars[a]:
  43. selector = vars[a]['kindselector']
  44. lk = 'kind'
  45. elif 'charselector' in vars[a]:
  46. selector = vars[a]['charselector']
  47. lk = 'len'
  48. if '*' in selector:
  49. if f90mode:
  50. if selector['*'] in ['*', ':', '(*)']:
  51. vardef = f'{vardef}(len=*)'
  52. else:
  53. vardef = f"{vardef}({lk}={selector['*']})"
  54. elif selector['*'] in ['*', ':']:
  55. vardef = f"{vardef}*({selector['*']})"
  56. else:
  57. vardef = f"{vardef}*{selector['*']}"
  58. elif 'len' in selector:
  59. vardef = f"{vardef}(len={selector['len']}"
  60. if 'kind' in selector:
  61. vardef = f"{vardef},kind={selector['kind']})"
  62. else:
  63. vardef = f'{vardef})'
  64. elif 'kind' in selector:
  65. vardef = f"{vardef}(kind={selector['kind']})"
  66. vardef = f'{vardef} {fa}'
  67. if 'dimension' in vars[a]:
  68. vardef = f"{vardef}({','.join(vars[a]['dimension'])})"
  69. return vardef
  70. def useiso_c_binding(rout):
  71. useisoc = False
  72. for value in rout['vars'].values():
  73. kind_value = value.get('kindselector', {}).get('kind')
  74. if kind_value in isoc_kindmap:
  75. return True
  76. return useisoc
  77. def createfuncwrapper(rout, signature=0):
  78. assert isfunction(rout)
  79. extra_args = []
  80. vars = rout['vars']
  81. for a in rout['args']:
  82. v = rout['vars'][a]
  83. for i, d in enumerate(v.get('dimension', [])):
  84. if d == ':':
  85. dn = f'f2py_{a}_d{i}'
  86. dv = {'typespec': 'integer', 'intent': ['hide']}
  87. dv['='] = f'shape({a}, {i})'
  88. extra_args.append(dn)
  89. vars[dn] = dv
  90. v['dimension'][i] = dn
  91. rout['args'].extend(extra_args)
  92. need_interface = bool(extra_args)
  93. ret = ['']
  94. def add(line, ret=ret):
  95. ret[0] = f'{ret[0]}\n {line}'
  96. name = rout['name']
  97. fortranname = getfortranname(rout)
  98. f90mode = ismoduleroutine(rout)
  99. newname = f'{name}f2pywrap'
  100. if newname not in vars:
  101. vars[newname] = vars[name]
  102. args = [newname] + rout['args'][1:]
  103. else:
  104. args = [newname] + rout['args']
  105. l_tmpl = var2fixfortran(vars, name, '@@@NAME@@@', f90mode)
  106. if l_tmpl[:13] == 'character*(*)':
  107. if f90mode:
  108. l_tmpl = 'character(len=10)' + l_tmpl[13:]
  109. else:
  110. l_tmpl = 'character*10' + l_tmpl[13:]
  111. charselect = vars[name]['charselector']
  112. if charselect.get('*', '') == '(*)':
  113. charselect['*'] = '10'
  114. l1 = l_tmpl.replace('@@@NAME@@@', newname)
  115. rl = None
  116. useisoc = useiso_c_binding(rout)
  117. sargs = ', '.join(args)
  118. if f90mode:
  119. # gh-23598 fix warning
  120. # Essentially, this gets called again with modules where the name of the
  121. # function is added to the arguments, which is not required, and removed
  122. sargs = sargs.replace(f"{name}, ", '')
  123. args = [arg for arg in args if arg != name]
  124. rout['args'] = args
  125. add(f"subroutine f2pywrap_{rout['modulename']}_{name} ({sargs})")
  126. if not signature:
  127. add(f"use {rout['modulename']}, only : {fortranname}")
  128. if useisoc:
  129. add('use iso_c_binding')
  130. else:
  131. add(f'subroutine f2pywrap{name} ({sargs})')
  132. if useisoc:
  133. add('use iso_c_binding')
  134. if not need_interface:
  135. add(f'external {fortranname}')
  136. rl = l_tmpl.replace('@@@NAME@@@', '') + ' ' + fortranname
  137. if need_interface:
  138. for line in rout['saved_interface'].split('\n'):
  139. if line.lstrip().startswith('use ') and '__user__' not in line:
  140. add(line)
  141. args = args[1:]
  142. dumped_args = []
  143. for a in args:
  144. if isexternal(vars[a]):
  145. add(f'external {a}')
  146. dumped_args.append(a)
  147. for a in args:
  148. if a in dumped_args:
  149. continue
  150. if isscalar(vars[a]):
  151. add(var2fixfortran(vars, a, f90mode=f90mode))
  152. dumped_args.append(a)
  153. for a in args:
  154. if a in dumped_args:
  155. continue
  156. if isintent_in(vars[a]):
  157. add(var2fixfortran(vars, a, f90mode=f90mode))
  158. dumped_args.append(a)
  159. for a in args:
  160. if a in dumped_args:
  161. continue
  162. add(var2fixfortran(vars, a, f90mode=f90mode))
  163. add(l1)
  164. if rl is not None:
  165. add(rl)
  166. if need_interface:
  167. if f90mode:
  168. # f90 module already defines needed interface
  169. pass
  170. else:
  171. add('interface')
  172. add(rout['saved_interface'].lstrip())
  173. add('end interface')
  174. sargs = ', '.join([a for a in args if a not in extra_args])
  175. if not signature:
  176. if islogicalfunction(rout):
  177. add(f'{newname} = .not.(.not.{fortranname}({sargs}))')
  178. else:
  179. add(f'{newname} = {fortranname}({sargs})')
  180. if f90mode:
  181. add(f"end subroutine f2pywrap_{rout['modulename']}_{name}")
  182. else:
  183. add('end')
  184. return ret[0]
  185. def createsubrwrapper(rout, signature=0):
  186. assert issubroutine(rout)
  187. extra_args = []
  188. vars = rout['vars']
  189. for a in rout['args']:
  190. v = rout['vars'][a]
  191. for i, d in enumerate(v.get('dimension', [])):
  192. if d == ':':
  193. dn = f'f2py_{a}_d{i}'
  194. dv = {'typespec': 'integer', 'intent': ['hide']}
  195. dv['='] = f'shape({a}, {i})'
  196. extra_args.append(dn)
  197. vars[dn] = dv
  198. v['dimension'][i] = dn
  199. rout['args'].extend(extra_args)
  200. need_interface = bool(extra_args)
  201. ret = ['']
  202. def add(line, ret=ret):
  203. ret[0] = f'{ret[0]}\n {line}'
  204. name = rout['name']
  205. fortranname = getfortranname(rout)
  206. f90mode = ismoduleroutine(rout)
  207. args = rout['args']
  208. useisoc = useiso_c_binding(rout)
  209. sargs = ', '.join(args)
  210. if f90mode:
  211. add(f"subroutine f2pywrap_{rout['modulename']}_{name} ({sargs})")
  212. if useisoc:
  213. add('use iso_c_binding')
  214. if not signature:
  215. add(f"use {rout['modulename']}, only : {fortranname}")
  216. else:
  217. add(f'subroutine f2pywrap{name} ({sargs})')
  218. if useisoc:
  219. add('use iso_c_binding')
  220. if not need_interface:
  221. add(f'external {fortranname}')
  222. if need_interface:
  223. for line in rout['saved_interface'].split('\n'):
  224. if line.lstrip().startswith('use ') and '__user__' not in line:
  225. add(line)
  226. dumped_args = []
  227. for a in args:
  228. if isexternal(vars[a]):
  229. add(f'external {a}')
  230. dumped_args.append(a)
  231. for a in args:
  232. if a in dumped_args:
  233. continue
  234. if isscalar(vars[a]):
  235. add(var2fixfortran(vars, a, f90mode=f90mode))
  236. dumped_args.append(a)
  237. for a in args:
  238. if a in dumped_args:
  239. continue
  240. add(var2fixfortran(vars, a, f90mode=f90mode))
  241. if need_interface:
  242. if f90mode:
  243. # f90 module already defines needed interface
  244. pass
  245. else:
  246. add('interface')
  247. for line in rout['saved_interface'].split('\n'):
  248. if line.lstrip().startswith('use ') and '__user__' in line:
  249. continue
  250. add(line)
  251. add('end interface')
  252. sargs = ', '.join([a for a in args if a not in extra_args])
  253. if not signature:
  254. add(f'call {fortranname}({sargs})')
  255. if f90mode:
  256. add(f"end subroutine f2pywrap_{rout['modulename']}_{name}")
  257. else:
  258. add('end')
  259. return ret[0]
  260. def assubr(rout):
  261. if isfunction_wrap(rout):
  262. fortranname = getfortranname(rout)
  263. name = rout['name']
  264. outmess('\t\tCreating wrapper for Fortran function "%s"("%s")...\n' % (
  265. name, fortranname))
  266. rout = copy.copy(rout)
  267. fname = name
  268. rname = fname
  269. if 'result' in rout:
  270. rname = rout['result']
  271. rout['vars'][fname] = rout['vars'][rname]
  272. fvar = rout['vars'][fname]
  273. if not isintent_out(fvar):
  274. if 'intent' not in fvar:
  275. fvar['intent'] = []
  276. fvar['intent'].append('out')
  277. flag = 1
  278. for i in fvar['intent']:
  279. if i.startswith('out='):
  280. flag = 0
  281. break
  282. if flag:
  283. fvar['intent'].append(f'out={rname}')
  284. rout['args'][:] = [fname] + rout['args']
  285. return rout, createfuncwrapper(rout)
  286. if issubroutine_wrap(rout):
  287. fortranname = getfortranname(rout)
  288. name = rout['name']
  289. outmess('\t\tCreating wrapper for Fortran subroutine "%s"("%s")...\n'
  290. % (name, fortranname))
  291. rout = copy.copy(rout)
  292. return rout, createsubrwrapper(rout)
  293. return rout, ''