xfys commited on
Commit
f07f089
1 Parent(s): a540323

Upload 432 files

Browse files
This view is limited to 50 files because it contains too many changes.   See raw diff
Files changed (50) hide show
  1. Cython-0.29.28.dist-info/COPYING.txt +19 -0
  2. Cython-0.29.28.dist-info/INSTALLER +1 -0
  3. Cython-0.29.28.dist-info/LICENSE.txt +176 -0
  4. Cython-0.29.28.dist-info/METADATA +60 -0
  5. Cython-0.29.28.dist-info/RECORD +434 -0
  6. Cython-0.29.28.dist-info/REQUESTED +0 -0
  7. Cython-0.29.28.dist-info/WHEEL +5 -0
  8. Cython-0.29.28.dist-info/direct_url.json +1 -0
  9. Cython-0.29.28.dist-info/entry_points.txt +5 -0
  10. Cython-0.29.28.dist-info/top_level.txt +3 -0
  11. Cython/Build/BuildExecutable.py +142 -0
  12. Cython/Build/Cythonize.py +229 -0
  13. Cython/Build/Dependencies.py +1308 -0
  14. Cython/Build/Distutils.py +1 -0
  15. Cython/Build/Inline.py +376 -0
  16. Cython/Build/IpythonMagic.py +565 -0
  17. Cython/Build/Tests/TestCyCache.py +106 -0
  18. Cython/Build/Tests/TestInline.py +96 -0
  19. Cython/Build/Tests/TestIpythonMagic.py +205 -0
  20. Cython/Build/Tests/TestStripLiterals.py +57 -0
  21. Cython/Build/Tests/__init__.py +1 -0
  22. Cython/Build/Tests/__pycache__/TestCyCache.cpython-39.pyc +0 -0
  23. Cython/Build/Tests/__pycache__/TestInline.cpython-39.pyc +0 -0
  24. Cython/Build/Tests/__pycache__/TestIpythonMagic.cpython-39.pyc +0 -0
  25. Cython/Build/Tests/__pycache__/TestStripLiterals.cpython-39.pyc +0 -0
  26. Cython/Build/Tests/__pycache__/__init__.cpython-39.pyc +0 -0
  27. Cython/Build/__init__.py +2 -0
  28. Cython/Build/__pycache__/BuildExecutable.cpython-39.pyc +0 -0
  29. Cython/Build/__pycache__/Cythonize.cpython-39.pyc +0 -0
  30. Cython/Build/__pycache__/Dependencies.cpython-39.pyc +0 -0
  31. Cython/Build/__pycache__/Distutils.cpython-39.pyc +0 -0
  32. Cython/Build/__pycache__/Inline.cpython-39.pyc +0 -0
  33. Cython/Build/__pycache__/IpythonMagic.cpython-39.pyc +0 -0
  34. Cython/Build/__pycache__/__init__.cpython-39.pyc +0 -0
  35. Cython/CodeWriter.py +816 -0
  36. Cython/Compiler/AnalysedTreeTransforms.py +99 -0
  37. Cython/Compiler/Annotate.py +317 -0
  38. Cython/Compiler/AutoDocTransforms.py +214 -0
  39. Cython/Compiler/Buffer.py +740 -0
  40. Cython/Compiler/Builtin.py +449 -0
  41. Cython/Compiler/CmdLine.py +224 -0
  42. Cython/Compiler/Code.pxd +124 -0
  43. Cython/Compiler/Code.py +2597 -0
  44. Cython/Compiler/CodeGeneration.py +35 -0
  45. Cython/Compiler/CythonScope.py +164 -0
  46. Cython/Compiler/DebugFlags.py +21 -0
  47. Cython/Compiler/Errors.py +265 -0
  48. Cython/Compiler/ExprNodes.py +0 -0
  49. Cython/Compiler/FlowControl.cp39-win_amd64.pyd +0 -0
  50. Cython/Compiler/FlowControl.pxd +111 -0
Cython-0.29.28.dist-info/COPYING.txt ADDED
@@ -0,0 +1,19 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ The original Pyrex code as of 2006-04 is licensed under the following
2
+ license: "Copyright stuff: Pyrex is free of restrictions. You may use,
3
+ redistribute, modify and distribute modified versions."
4
+
5
+ ------------------
6
+
7
+ Cython, which derives from Pyrex, is licensed under the Apache 2.0
8
+ Software License. More precisely, all modifications and new code
9
+ made to go from Pyrex to Cython are so licensed.
10
+
11
+ See LICENSE.txt for more details.
12
+
13
+ ------------------
14
+
15
+ The output of a Cython compilation is NOT considered a derivative
16
+ work of Cython. Specifically, though the compilation process may
17
+ embed snippets of varying lengths into the final output, these
18
+ snippets, as embedded in the output, do not encumber the resulting
19
+ output with any license restrictions.
Cython-0.29.28.dist-info/INSTALLER ADDED
@@ -0,0 +1 @@
 
 
1
+ conda
Cython-0.29.28.dist-info/LICENSE.txt ADDED
@@ -0,0 +1,176 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction,
10
+ and distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by
13
+ the copyright owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all
16
+ other entities that control, are controlled by, or are under common
17
+ control with that entity. For the purposes of this definition,
18
+ "control" means (i) the power, direct or indirect, to cause the
19
+ direction or management of such entity, whether by contract or
20
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
21
+ outstanding shares, or (iii) beneficial ownership of such entity.
22
+
23
+ "You" (or "Your") shall mean an individual or Legal Entity
24
+ exercising permissions granted by this License.
25
+
26
+ "Source" form shall mean the preferred form for making modifications,
27
+ including but not limited to software source code, documentation
28
+ source, and configuration files.
29
+
30
+ "Object" form shall mean any form resulting from mechanical
31
+ transformation or translation of a Source form, including but
32
+ not limited to compiled object code, generated documentation,
33
+ and conversions to other media types.
34
+
35
+ "Work" shall mean the work of authorship, whether in Source or
36
+ Object form, made available under the License, as indicated by a
37
+ copyright notice that is included in or attached to the work
38
+ (an example is provided in the Appendix below).
39
+
40
+ "Derivative Works" shall mean any work, whether in Source or Object
41
+ form, that is based on (or derived from) the Work and for which the
42
+ editorial revisions, annotations, elaborations, or other modifications
43
+ represent, as a whole, an original work of authorship. For the purposes
44
+ of this License, Derivative Works shall not include works that remain
45
+ separable from, or merely link (or bind by name) to the interfaces of,
46
+ the Work and Derivative Works thereof.
47
+
48
+ "Contribution" shall mean any work of authorship, including
49
+ the original version of the Work and any modifications or additions
50
+ to that Work or Derivative Works thereof, that is intentionally
51
+ submitted to Licensor for inclusion in the Work by the copyright owner
52
+ or by an individual or Legal Entity authorized to submit on behalf of
53
+ the copyright owner. For the purposes of this definition, "submitted"
54
+ means any form of electronic, verbal, or written communication sent
55
+ to the Licensor or its representatives, including but not limited to
56
+ communication on electronic mailing lists, source code control systems,
57
+ and issue tracking systems that are managed by, or on behalf of, the
58
+ Licensor for the purpose of discussing and improving the Work, but
59
+ excluding communication that is conspicuously marked or otherwise
60
+ designated in writing by the copyright owner as "Not a Contribution."
61
+
62
+ "Contributor" shall mean Licensor and any individual or Legal Entity
63
+ on behalf of whom a Contribution has been received by Licensor and
64
+ subsequently incorporated within the Work.
65
+
66
+ 2. Grant of Copyright License. Subject to the terms and conditions of
67
+ this License, each Contributor hereby grants to You a perpetual,
68
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69
+ copyright license to reproduce, prepare Derivative Works of,
70
+ publicly display, publicly perform, sublicense, and distribute the
71
+ Work and such Derivative Works in Source or Object form.
72
+
73
+ 3. Grant of Patent License. Subject to the terms and conditions of
74
+ this License, each Contributor hereby grants to You a perpetual,
75
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76
+ (except as stated in this section) patent license to make, have made,
77
+ use, offer to sell, sell, import, and otherwise transfer the Work,
78
+ where such license applies only to those patent claims licensable
79
+ by such Contributor that are necessarily infringed by their
80
+ Contribution(s) alone or by combination of their Contribution(s)
81
+ with the Work to which such Contribution(s) was submitted. If You
82
+ institute patent litigation against any entity (including a
83
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
84
+ or a Contribution incorporated within the Work constitutes direct
85
+ or contributory patent infringement, then any patent licenses
86
+ granted to You under this License for that Work shall terminate
87
+ as of the date such litigation is filed.
88
+
89
+ 4. Redistribution. You may reproduce and distribute copies of the
90
+ Work or Derivative Works thereof in any medium, with or without
91
+ modifications, and in Source or Object form, provided that You
92
+ meet the following conditions:
93
+
94
+ (a) You must give any other recipients of the Work or
95
+ Derivative Works a copy of this License; and
96
+
97
+ (b) You must cause any modified files to carry prominent notices
98
+ stating that You changed the files; and
99
+
100
+ (c) You must retain, in the Source form of any Derivative Works
101
+ that You distribute, all copyright, patent, trademark, and
102
+ attribution notices from the Source form of the Work,
103
+ excluding those notices that do not pertain to any part of
104
+ the Derivative Works; and
105
+
106
+ (d) If the Work includes a "NOTICE" text file as part of its
107
+ distribution, then any Derivative Works that You distribute must
108
+ include a readable copy of the attribution notices contained
109
+ within such NOTICE file, excluding those notices that do not
110
+ pertain to any part of the Derivative Works, in at least one
111
+ of the following places: within a NOTICE text file distributed
112
+ as part of the Derivative Works; within the Source form or
113
+ documentation, if provided along with the Derivative Works; or,
114
+ within a display generated by the Derivative Works, if and
115
+ wherever such third-party notices normally appear. The contents
116
+ of the NOTICE file are for informational purposes only and
117
+ do not modify the License. You may add Your own attribution
118
+ notices within Derivative Works that You distribute, alongside
119
+ or as an addendum to the NOTICE text from the Work, provided
120
+ that such additional attribution notices cannot be construed
121
+ as modifying the License.
122
+
123
+ You may add Your own copyright statement to Your modifications and
124
+ may provide additional or different license terms and conditions
125
+ for use, reproduction, or distribution of Your modifications, or
126
+ for any such Derivative Works as a whole, provided Your use,
127
+ reproduction, and distribution of the Work otherwise complies with
128
+ the conditions stated in this License.
129
+
130
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
131
+ any Contribution intentionally submitted for inclusion in the Work
132
+ by You to the Licensor shall be under the terms and conditions of
133
+ this License, without any additional terms or conditions.
134
+ Notwithstanding the above, nothing herein shall supersede or modify
135
+ the terms of any separate license agreement you may have executed
136
+ with Licensor regarding such Contributions.
137
+
138
+ 6. Trademarks. This License does not grant permission to use the trade
139
+ names, trademarks, service marks, or product names of the Licensor,
140
+ except as required for reasonable and customary use in describing the
141
+ origin of the Work and reproducing the content of the NOTICE file.
142
+
143
+ 7. Disclaimer of Warranty. Unless required by applicable law or
144
+ agreed to in writing, Licensor provides the Work (and each
145
+ Contributor provides its Contributions) on an "AS IS" BASIS,
146
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147
+ implied, including, without limitation, any warranties or conditions
148
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149
+ PARTICULAR PURPOSE. You are solely responsible for determining the
150
+ appropriateness of using or redistributing the Work and assume any
151
+ risks associated with Your exercise of permissions under this License.
152
+
153
+ 8. Limitation of Liability. In no event and under no legal theory,
154
+ whether in tort (including negligence), contract, or otherwise,
155
+ unless required by applicable law (such as deliberate and grossly
156
+ negligent acts) or agreed to in writing, shall any Contributor be
157
+ liable to You for damages, including any direct, indirect, special,
158
+ incidental, or consequential damages of any character arising as a
159
+ result of this License or out of the use or inability to use the
160
+ Work (including but not limited to damages for loss of goodwill,
161
+ work stoppage, computer failure or malfunction, or any and all
162
+ other commercial damages or losses), even if such Contributor
163
+ has been advised of the possibility of such damages.
164
+
165
+ 9. Accepting Warranty or Additional Liability. While redistributing
166
+ the Work or Derivative Works thereof, You may choose to offer,
167
+ and charge a fee for, acceptance of support, warranty, indemnity,
168
+ or other liability obligations and/or rights consistent with this
169
+ License. However, in accepting such obligations, You may act only
170
+ on Your own behalf and on Your sole responsibility, not on behalf
171
+ of any other Contributor, and only if You agree to indemnify,
172
+ defend, and hold each Contributor harmless for any liability
173
+ incurred by, or claims asserted against, such Contributor by reason
174
+ of your accepting any such warranty or additional liability.
175
+
176
+ END OF TERMS AND CONDITIONS
Cython-0.29.28.dist-info/METADATA ADDED
@@ -0,0 +1,60 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ Metadata-Version: 2.1
2
+ Name: Cython
3
+ Version: 0.29.28
4
+ Summary: The Cython compiler for writing C extensions for the Python language.
5
+ Home-page: http://cython.org/
6
+ Author: Robert Bradshaw, Stefan Behnel, Dag Seljebotn, Greg Ewing, et al.
7
+ Author-email: cython-devel@python.org
8
+ License: Apache
9
+ Platform: UNKNOWN
10
+ Classifier: Development Status :: 5 - Production/Stable
11
+ Classifier: Intended Audience :: Developers
12
+ Classifier: License :: OSI Approved :: Apache Software License
13
+ Classifier: Operating System :: OS Independent
14
+ Classifier: Programming Language :: Python
15
+ Classifier: Programming Language :: Python :: 2
16
+ Classifier: Programming Language :: Python :: 2.6
17
+ Classifier: Programming Language :: Python :: 2.7
18
+ Classifier: Programming Language :: Python :: 3
19
+ Classifier: Programming Language :: Python :: 3.4
20
+ Classifier: Programming Language :: Python :: 3.5
21
+ Classifier: Programming Language :: Python :: 3.6
22
+ Classifier: Programming Language :: Python :: 3.7
23
+ Classifier: Programming Language :: Python :: 3.8
24
+ Classifier: Programming Language :: Python :: 3.9
25
+ Classifier: Programming Language :: Python :: 3.10
26
+ Classifier: Programming Language :: Python :: Implementation :: CPython
27
+ Classifier: Programming Language :: Python :: Implementation :: PyPy
28
+ Classifier: Programming Language :: C
29
+ Classifier: Programming Language :: Cython
30
+ Classifier: Topic :: Software Development :: Code Generators
31
+ Classifier: Topic :: Software Development :: Compilers
32
+ Classifier: Topic :: Software Development :: Libraries :: Python Modules
33
+ Requires-Python: >=2.6, !=3.0.*, !=3.1.*, !=3.2.*
34
+ License-File: LICENSE.txt
35
+ License-File: COPYING.txt
36
+
37
+ The Cython language makes writing C extensions for the Python language as
38
+ easy as Python itself. Cython is a source code translator based on Pyrex_,
39
+ but supports more cutting edge functionality and optimizations.
40
+
41
+ The Cython language is a superset of the Python language (almost all Python
42
+ code is also valid Cython code), but Cython additionally supports optional
43
+ static typing to natively call C functions, operate with C++ classes and
44
+ declare fast C types on variables and class attributes. This allows the
45
+ compiler to generate very efficient C code from Cython code.
46
+
47
+ This makes Cython the ideal language for writing glue code for external
48
+ C/C++ libraries, and for fast C modules that speed up the execution of
49
+ Python code.
50
+
51
+ Note that for one-time builds, e.g. for CI/testing, on platforms that are not
52
+ covered by one of the wheel packages provided on PyPI *and* the pure Python wheel
53
+ that we provide is not used, it is substantially faster than a full source build
54
+ to install an uncompiled (slower) version of Cython with::
55
+
56
+ pip install Cython --install-option="--no-cython-compile"
57
+
58
+ .. _Pyrex: http://www.cosc.canterbury.ac.nz/greg.ewing/python/Pyrex/
59
+
60
+
Cython-0.29.28.dist-info/RECORD ADDED
@@ -0,0 +1,434 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ ../../Scripts/cygdb.exe,sha256=ygPa8EZCtjHaHPKSztwcdtZXRRP0VTJbA5BK1Z8Zhvw,106354
2
+ ../../Scripts/cython.exe,sha256=CtTvOjOheysEFKXIFwX1MX6o75YgMNqbvuLDUVWgZps,106375
3
+ ../../Scripts/cythonize.exe,sha256=RH77nH_tm3CHUGc1B9eUiHYvPAXxPt3yerc-HX9eF6w,106355
4
+ Cython-0.29.28.dist-info/COPYING.txt,sha256=4escSahQjoFz2sMBV-SmQ5pErYhGGUdGxCT7w_wrldc,756
5
+ Cython-0.29.28.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
6
+ Cython-0.29.28.dist-info/LICENSE.txt,sha256=psuoW8kuDP96RQsdhzwOqi6fyWv0ct8CR6Jr7He_P_k,10173
7
+ Cython-0.29.28.dist-info/METADATA,sha256=0XNJlufb1avaP2cLvsCbMTto2rcey0ofqpsXR3_cGiI,2787
8
+ Cython-0.29.28.dist-info/RECORD,,
9
+ Cython-0.29.28.dist-info/REQUESTED,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
10
+ Cython-0.29.28.dist-info/WHEEL,sha256=fVcVlLzi8CGi_Ul8vjMdn8gER25dn5GBg9E6k9z41-Y,100
11
+ Cython-0.29.28.dist-info/direct_url.json,sha256=YwlnxYRYlewtshuCDQgo1231pj9uP-3ff2opeCNOfFs,66
12
+ Cython-0.29.28.dist-info/entry_points.txt,sha256=2BOm1wPEnW-cqvf6H044-DF64esaA6qeqkbDlSvz1as,140
13
+ Cython-0.29.28.dist-info/top_level.txt,sha256=jLV8tZV98iCbIfiJR4DVzTX5Ru1Y_pYMZ59wkMCe6SY,24
14
+ Cython/Build/BuildExecutable.py,sha256=9wjcOncQpbCT_Pu2Ljj5jufFLSY6T_oHOFt0uXYTdRk,4318
15
+ Cython/Build/Cythonize.py,sha256=OBSutgd93PXiRi5_6GRuhpfMZehw9YOfRgt1mxi6QLk,8359
16
+ Cython/Build/Dependencies.py,sha256=SkMflfSEXLlyGdkQ0xLMY7zMMoBxq4VgfPyN_BUSvTQ,50391
17
+ Cython/Build/Distutils.py,sha256=iO5tPX84Kc-ZWMocfuQbl_PqyC9HGGIRS-NiKI60-ZE,49
18
+ Cython/Build/Inline.py,sha256=wd5_xBOup8MWxNtv5uwWgWWG5GrkShYysMMluY7ripE,13451
19
+ Cython/Build/IpythonMagic.py,sha256=j-E-JJYGj03ceWDp5SRmyK-WxyroRkNOEQODA9rCfFc,21126
20
+ Cython/Build/Tests/TestCyCache.py,sha256=olOvphv4q1CLsNzMAhgmXnL77zhGGQKuKbYs_iSFFvA,4151
21
+ Cython/Build/Tests/TestInline.py,sha256=bp4XGXZYSyxY2zXI0q4bg58a0ARsclUzXSfWykn0dZw,2854
22
+ Cython/Build/Tests/TestIpythonMagic.py,sha256=H88J6_r3dP5V3-N7eSKZuE2l9G8BtkFF_o8wY7pifKU,6225
23
+ Cython/Build/Tests/TestStripLiterals.py,sha256=-QeUd22OnoL50rW2EgpfIA01UzRMutcBA5NrhkHiE7M,1550
24
+ Cython/Build/Tests/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
25
+ Cython/Build/Tests/__pycache__/TestCyCache.cpython-39.pyc,,
26
+ Cython/Build/Tests/__pycache__/TestInline.cpython-39.pyc,,
27
+ Cython/Build/Tests/__pycache__/TestIpythonMagic.cpython-39.pyc,,
28
+ Cython/Build/Tests/__pycache__/TestStripLiterals.cpython-39.pyc,,
29
+ Cython/Build/Tests/__pycache__/__init__.cpython-39.pyc,,
30
+ Cython/Build/__init__.py,sha256=zBhW6hT9Mwk1ZybfuPi61iCa6A4srId1HJz9OiEd07o,69
31
+ Cython/Build/__pycache__/BuildExecutable.cpython-39.pyc,,
32
+ Cython/Build/__pycache__/Cythonize.cpython-39.pyc,,
33
+ Cython/Build/__pycache__/Dependencies.cpython-39.pyc,,
34
+ Cython/Build/__pycache__/Distutils.cpython-39.pyc,,
35
+ Cython/Build/__pycache__/Inline.cpython-39.pyc,,
36
+ Cython/Build/__pycache__/IpythonMagic.cpython-39.pyc,,
37
+ Cython/Build/__pycache__/__init__.cpython-39.pyc,,
38
+ Cython/CodeWriter.py,sha256=Sa1hLmUcIMnCGTNASqyzfGqk70i6v-YslD5ogllCaiY,23901
39
+ Cython/Compiler/AnalysedTreeTransforms.py,sha256=T2r1SLFeG7a4D9tt93hm8YRH_aGLwmK82PNpb1RsbnE,3826
40
+ Cython/Compiler/Annotate.py,sha256=bkVgdH3ItuIbaSpi9Qo5bKiIjOX3-J5sZORfcyk_eZY,12950
41
+ Cython/Compiler/AutoDocTransforms.py,sha256=le7k-xhGWDkvpRXRppZsqhJzkav6i41GmAOmDetxVmk,7517
42
+ Cython/Compiler/Buffer.py,sha256=6-YIUiLWtgpC2LRhs7116umzNiEbzA6cz1lnPkSEn30,29035
43
+ Cython/Compiler/Builtin.py,sha256=yuMN5KlryVwMR02wVwMcdgQXXbIy14yQaL-dw1aVTeE,22989
44
+ Cython/Compiler/CmdLine.py,sha256=cRwX-QUfBCVmGXxVKKAeF2CHKiLVtjH4QRGMGaBRmkI,9939
45
+ Cython/Compiler/Code.pxd,sha256=XEZU44jXCOFoYTG15-R0X41krTrjpuo1wxRKjyzYwoY,3354
46
+ Cython/Compiler/Code.py,sha256=N7qoQuOnXc3HXlY0spOP-d0LY8WCsU2vRaV7-MNSG6c,97198
47
+ Cython/Compiler/CodeGeneration.py,sha256=jkcx2uX07nck0UZSgysIThRuJiPbdkSeXR4Z2uzbQU8,1108
48
+ Cython/Compiler/CythonScope.py,sha256=mNwmE509uePmR3S2djg3Dq6zOZ3hgK-U8NDeawys9WM,6027
49
+ Cython/Compiler/DebugFlags.py,sha256=5Zg9ETp0qPFEma6QMtrGUwu9Fn6NTYMBMWPI_GxFW0A,623
50
+ Cython/Compiler/Errors.py,sha256=GATz9x6onls09cM6TeDw3kdBgdCxUiKJBILwukBF6WI,7554
51
+ Cython/Compiler/ExprNodes.py,sha256=qsOOc1fQyPtGp89dTTnmfXaO_KppGtaCdw1Ljh47Vo4,549114
52
+ Cython/Compiler/FlowControl.cp39-win_amd64.pyd,sha256=3t3R4866IlXDucRNsciEko9a-_FfvJbQJKxIOyKqVvs,450048
53
+ Cython/Compiler/FlowControl.pxd,sha256=W8bqGCJLzvAhnL3d1OF8798ZDJg0QI0eA_ebnA4dkoQ,2918
54
+ Cython/Compiler/FlowControl.py,sha256=-I33Yd9sp1RCENjFkrufBM8Xrj3y5SI7KZ04Vz5Djiw,45981
55
+ Cython/Compiler/FusedNode.cp39-win_amd64.pyd,sha256=X-P965DwQD-eiX5hZoFJMFkCs7wFbGpbOces8wsxFXI,267776
56
+ Cython/Compiler/FusedNode.py,sha256=qmHVHylEPpllK_x-471xBG-zMzAt0RF52tLxt_-RZqs,37794
57
+ Cython/Compiler/Future.py,sha256=GwcWZ_Vti0atfbOARfS2kIvZOvRuPu38wbShIn4o4kA,587
58
+ Cython/Compiler/Interpreter.py,sha256=iNweexX2HDI5nZj2rzkW-lw9Rq3gzM__P7SBqH3uxbU,2106
59
+ Cython/Compiler/Lexicon.py,sha256=Cw_wIfQymcTEdkoo82V2xbV8kvCp30O-Pc7qF4hbfCI,4855
60
+ Cython/Compiler/Main.py,sha256=_b1cDedgPRDFy4sN7-FZsKx5qbcD4XTefhD9qZwlgWQ,36488
61
+ Cython/Compiler/MemoryView.py,sha256=c6J7PtQ6wccb9uBxvbLngia4jO-h2uea7viIzJNhDYU,30009
62
+ Cython/Compiler/ModuleNode.py,sha256=n2s-hf3IegF4n32np8l8Lg-EBPNK39K1_FmNdqRdJk4,142244
63
+ Cython/Compiler/Naming.py,sha256=Z6FLRoOl21F91SMoIoMSu_CqYosGhxDJLO9grYSHhVI,6333
64
+ Cython/Compiler/Nodes.py,sha256=ydfZDpwwfLKZFDoTbk6LV66-R80GLfeCUiXo9BDljOo,391269
65
+ Cython/Compiler/Optimize.py,sha256=W01lBStxLyXYYRhBUZrr-PZYyfyl9KgY3KPvriVXf8U,210273
66
+ Cython/Compiler/Options.py,sha256=ryrLc_NNo0QSMaP-jHhmo1L2oZJA6h7KpE6xs_KSyUo,19707
67
+ Cython/Compiler/ParseTreeTransforms.pxd,sha256=oDSda3XYP79o8tCAxEm_epWWVPXSaPy8lYrprWYRyNk,2468
68
+ Cython/Compiler/ParseTreeTransforms.py,sha256=TBKwvWfsV_f74ra_lVPkb3ETOs5_com8ZLAqqERrF8o,138431
69
+ Cython/Compiler/Parsing.pxd,sha256=pL_EQdVWaw7EZVTZrxzMcUvoSJeAoXXPRh8kCLudysk,8984
70
+ Cython/Compiler/Parsing.py,sha256=u_5SWON1ws-S_5cCw1xXmZdgxTlLWfTxFAJJEbXX7m8,129791
71
+ Cython/Compiler/Pipeline.py,sha256=6ravd0QCR5sCoKlz9HEz209A2UqgLp4Qp0VysoKa_mI,14061
72
+ Cython/Compiler/PyrexTypes.py,sha256=-nGJxm5amgIjkfvBpQFeSLn8zS2fi_cBvOtgBIfr5F8,173958
73
+ Cython/Compiler/Pythran.py,sha256=NHIml0yx0jPLyTLRAHXZr0LHTyEyfYqspgYuV4vdNKI,7267
74
+ Cython/Compiler/Scanning.cp39-win_amd64.pyd,sha256=5-LsBGSjEb90oBHZgigU6zrNGORvcYbFxPO3aaMeC4w,212480
75
+ Cython/Compiler/Scanning.pxd,sha256=vjjPLZb5udPzMpk67DKojTTDUl31QU86oXyAMks7Hsw,2113
76
+ Cython/Compiler/Scanning.py,sha256=Gl7sU5rI-5H5v8z9QLZBh1hivS2cOGa9H878QdEpUU4,18438
77
+ Cython/Compiler/StringEncoding.py,sha256=dn3jVSL4JScbeYf1f56j5RJR9M58AnCDdJObC-cc3fg,10812
78
+ Cython/Compiler/Symtab.py,sha256=n0tdTnbPkSuU5EgPkr6nGBN98Lhbklbjt8o6NUFt264,111727
79
+ Cython/Compiler/Tests/TestBuffer.py,sha256=yw-KUu2pu4CnsqYaqxj5rpyYodmjJSEqUvQNxLwPL8I,4155
80
+ Cython/Compiler/Tests/TestCmdLine.py,sha256=wBtbNserRsU4_Yj_0W6cujG-07PkwORT-r1Z3oLHmcU,4414
81
+ Cython/Compiler/Tests/TestFlowControl.py,sha256=ge3iqBor6xe5MLaLbOtw7ETntJnAh8EequF1aetVzMw,1848
82
+ Cython/Compiler/Tests/TestGrammar.py,sha256=zWMvYG19nIH85Le8ragXt2vLBlWlGGNeMgrTdQO5JGM,3443
83
+ Cython/Compiler/Tests/TestMemView.py,sha256=yBAQ5tN8DVPTFRJ81dAzOepCt5Ly6fiaAssQve5ryy4,2515
84
+ Cython/Compiler/Tests/TestParseTreeTransforms.py,sha256=CorKAgH9tdoehstG6vBAbdpUhaPdN8EW277KMWNR4Uw,8503
85
+ Cython/Compiler/Tests/TestSignatureMatching.py,sha256=qMiQZeg5_Eu8VfCY_lMawqpjpKNV0r6p6-9czKec1aY,3338
86
+ Cython/Compiler/Tests/TestStringEncoding.py,sha256=RL1YDXrOUe1sPLEbWmTJQ5VF-uEZ_KLz0jaeQoMx85k,2315
87
+ Cython/Compiler/Tests/TestTreeFragment.py,sha256=wHlnF0ApwxeITx9pzg46P9N_2lM-7mrPwhDeNlIQnXM,2206
88
+ Cython/Compiler/Tests/TestTreePath.py,sha256=x-2KBIhSE6-vT-2BPe2q-zaa1oHtc42ibKzVs_y8_So,4238
89
+ Cython/Compiler/Tests/TestTypes.py,sha256=YuFib5WCJfSPafrhy5yrCUdwajYw61yGPo4HveTyzUs,669
90
+ Cython/Compiler/Tests/TestUtilityLoad.py,sha256=Uzf4_bOjha-zwQaikNbsAOVQs3ZPX3YD7QQ5T4s66YY,3341
91
+ Cython/Compiler/Tests/TestVisitor.py,sha256=QAnBpUhnirSFKqXWiawo-OhXhxIRTQidWxEzGjJDz6M,2228
92
+ Cython/Compiler/Tests/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
93
+ Cython/Compiler/Tests/__pycache__/TestBuffer.cpython-39.pyc,,
94
+ Cython/Compiler/Tests/__pycache__/TestCmdLine.cpython-39.pyc,,
95
+ Cython/Compiler/Tests/__pycache__/TestFlowControl.cpython-39.pyc,,
96
+ Cython/Compiler/Tests/__pycache__/TestGrammar.cpython-39.pyc,,
97
+ Cython/Compiler/Tests/__pycache__/TestMemView.cpython-39.pyc,,
98
+ Cython/Compiler/Tests/__pycache__/TestParseTreeTransforms.cpython-39.pyc,,
99
+ Cython/Compiler/Tests/__pycache__/TestSignatureMatching.cpython-39.pyc,,
100
+ Cython/Compiler/Tests/__pycache__/TestStringEncoding.cpython-39.pyc,,
101
+ Cython/Compiler/Tests/__pycache__/TestTreeFragment.cpython-39.pyc,,
102
+ Cython/Compiler/Tests/__pycache__/TestTreePath.cpython-39.pyc,,
103
+ Cython/Compiler/Tests/__pycache__/TestTypes.cpython-39.pyc,,
104
+ Cython/Compiler/Tests/__pycache__/TestUtilityLoad.cpython-39.pyc,,
105
+ Cython/Compiler/Tests/__pycache__/TestVisitor.cpython-39.pyc,,
106
+ Cython/Compiler/Tests/__pycache__/__init__.cpython-39.pyc,,
107
+ Cython/Compiler/TreeFragment.py,sha256=jQn4Lp2dNddJ-tjPquoFcyTcX9EIuTAbZKZAKs9-cGU,9408
108
+ Cython/Compiler/TreePath.py,sha256=3_lScMAd2Sly2ekZ8HO8dyZstGSruINl2MXXq9OYd2Q,7641
109
+ Cython/Compiler/TypeInference.py,sha256=s-GKZcq16KPPgY_OpF8cTlQmX1Cpu-qBMCtmAYDg8fc,22326
110
+ Cython/Compiler/TypeSlots.py,sha256=C2_eWwtL3nOnVCUX6LF9K9S6S2AawjeFMZryVtFq7TI,37306
111
+ Cython/Compiler/UtilNodes.py,sha256=mS6jlZ530p17WGU0ApbwvLecuByT18LFipVrKJg5jrM,11636
112
+ Cython/Compiler/UtilityCode.py,sha256=PbQtJt9fSwgm5xeXgYWQih6eUSmJL_RwTxOa5T9SrZU,9391
113
+ Cython/Compiler/Version.py,sha256=f2mS6aYYdu0DMRK3B4IuzMlCo-k-ffmehCao_vKlTdk,181
114
+ Cython/Compiler/Visitor.cp39-win_amd64.pyd,sha256=Nci2nH5xr3dcMCJNUA5H2q_54-wCfyDobaJsO53IGnc,242176
115
+ Cython/Compiler/Visitor.pxd,sha256=KvOZgHoEREMTVYXr1ZoAk9H4n__rpmhIwE2S11ajeYM,1792
116
+ Cython/Compiler/Visitor.py,sha256=iqtIsNaQwk8lSa6g_LnEU06CZtdnP7MDXbyodl4Ouwk,29984
117
+ Cython/Compiler/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
118
+ Cython/Compiler/__pycache__/AnalysedTreeTransforms.cpython-39.pyc,,
119
+ Cython/Compiler/__pycache__/Annotate.cpython-39.pyc,,
120
+ Cython/Compiler/__pycache__/AutoDocTransforms.cpython-39.pyc,,
121
+ Cython/Compiler/__pycache__/Buffer.cpython-39.pyc,,
122
+ Cython/Compiler/__pycache__/Builtin.cpython-39.pyc,,
123
+ Cython/Compiler/__pycache__/CmdLine.cpython-39.pyc,,
124
+ Cython/Compiler/__pycache__/Code.cpython-39.pyc,,
125
+ Cython/Compiler/__pycache__/CodeGeneration.cpython-39.pyc,,
126
+ Cython/Compiler/__pycache__/CythonScope.cpython-39.pyc,,
127
+ Cython/Compiler/__pycache__/DebugFlags.cpython-39.pyc,,
128
+ Cython/Compiler/__pycache__/Errors.cpython-39.pyc,,
129
+ Cython/Compiler/__pycache__/ExprNodes.cpython-39.pyc,,
130
+ Cython/Compiler/__pycache__/FlowControl.cpython-39.pyc,,
131
+ Cython/Compiler/__pycache__/FusedNode.cpython-39.pyc,,
132
+ Cython/Compiler/__pycache__/Future.cpython-39.pyc,,
133
+ Cython/Compiler/__pycache__/Interpreter.cpython-39.pyc,,
134
+ Cython/Compiler/__pycache__/Lexicon.cpython-39.pyc,,
135
+ Cython/Compiler/__pycache__/Main.cpython-39.pyc,,
136
+ Cython/Compiler/__pycache__/MemoryView.cpython-39.pyc,,
137
+ Cython/Compiler/__pycache__/ModuleNode.cpython-39.pyc,,
138
+ Cython/Compiler/__pycache__/Naming.cpython-39.pyc,,
139
+ Cython/Compiler/__pycache__/Nodes.cpython-39.pyc,,
140
+ Cython/Compiler/__pycache__/Optimize.cpython-39.pyc,,
141
+ Cython/Compiler/__pycache__/Options.cpython-39.pyc,,
142
+ Cython/Compiler/__pycache__/ParseTreeTransforms.cpython-39.pyc,,
143
+ Cython/Compiler/__pycache__/Parsing.cpython-39.pyc,,
144
+ Cython/Compiler/__pycache__/Pipeline.cpython-39.pyc,,
145
+ Cython/Compiler/__pycache__/PyrexTypes.cpython-39.pyc,,
146
+ Cython/Compiler/__pycache__/Pythran.cpython-39.pyc,,
147
+ Cython/Compiler/__pycache__/Scanning.cpython-39.pyc,,
148
+ Cython/Compiler/__pycache__/StringEncoding.cpython-39.pyc,,
149
+ Cython/Compiler/__pycache__/Symtab.cpython-39.pyc,,
150
+ Cython/Compiler/__pycache__/TreeFragment.cpython-39.pyc,,
151
+ Cython/Compiler/__pycache__/TreePath.cpython-39.pyc,,
152
+ Cython/Compiler/__pycache__/TypeInference.cpython-39.pyc,,
153
+ Cython/Compiler/__pycache__/TypeSlots.cpython-39.pyc,,
154
+ Cython/Compiler/__pycache__/UtilNodes.cpython-39.pyc,,
155
+ Cython/Compiler/__pycache__/UtilityCode.cpython-39.pyc,,
156
+ Cython/Compiler/__pycache__/Version.cpython-39.pyc,,
157
+ Cython/Compiler/__pycache__/Visitor.cpython-39.pyc,,
158
+ Cython/Compiler/__pycache__/__init__.cpython-39.pyc,,
159
+ Cython/Coverage.py,sha256=FtCMjKLYWvtULxWIzN-y3RhbwQwL4GtLyIjj-B3-07E,13537
160
+ Cython/Debugger/Cygdb.py,sha256=CH_pXm0Jhl4SAe6sJXa5NS47vMmQ2KBbecyV56vLqFE,5751
161
+ Cython/Debugger/DebugWriter.py,sha256=Yzz28JR4qZepxvxeu_1rJxIjJ4JbNQm5vM5e_UtNuRo,1945
162
+ Cython/Debugger/Tests/TestLibCython.py,sha256=xrENLEahnp6WtOfokVtsALR6Ot2jFR6T5ZZRcaX0Vxk,8327
163
+ Cython/Debugger/Tests/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
164
+ Cython/Debugger/Tests/__pycache__/TestLibCython.cpython-39.pyc,,
165
+ Cython/Debugger/Tests/__pycache__/__init__.cpython-39.pyc,,
166
+ Cython/Debugger/Tests/__pycache__/test_libcython_in_gdb.cpython-39.pyc,,
167
+ Cython/Debugger/Tests/__pycache__/test_libpython_in_gdb.cpython-39.pyc,,
168
+ Cython/Debugger/Tests/cfuncs.c,sha256=4SZurmnz5J1SiIs9N26Eu4zc2wvF_qMEKaN0eTcbDPo,71
169
+ Cython/Debugger/Tests/codefile,sha256=ugwpT9GPtYZIKe2Xco4PqikyA-poQAeYfE0icXmfb44,641
170
+ Cython/Debugger/Tests/test_libcython_in_gdb.py,sha256=EvPTYkd7nzR3JtFim-ASLI6wfcYhgWfI4BQrucLJbHY,15804
171
+ Cython/Debugger/Tests/test_libpython_in_gdb.py,sha256=1BD_FtMkmS4SoSQZq7MgAgDnvqIw3EcYZFVrtoCQmxo,4079
172
+ Cython/Debugger/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
173
+ Cython/Debugger/__pycache__/Cygdb.cpython-39.pyc,,
174
+ Cython/Debugger/__pycache__/DebugWriter.cpython-39.pyc,,
175
+ Cython/Debugger/__pycache__/__init__.cpython-39.pyc,,
176
+ Cython/Debugger/__pycache__/libcython.cpython-39.pyc,,
177
+ Cython/Debugger/__pycache__/libpython.cpython-39.pyc,,
178
+ Cython/Debugger/libcython.py,sha256=Qs0qGzeUyeY___3jRDy_WWIDFGfSRH4al7On2XxkuNg,44949
179
+ Cython/Debugger/libpython.py,sha256=IyTEdtGLnpQmt2XPgZ7oskQ8qGWWV2_5TMgZ5NhTA0k,90489
180
+ Cython/Debugging.py,sha256=vFtJhn7QstMf5gnYru2qHIz5ZjPg1KSlZVGHr-pBCwM,552
181
+ Cython/Distutils/__init__.py,sha256=uyWaN2NJ_mKYLzVsDPi0qZCdIYoW5M_7YYEmAOIL3Ek,98
182
+ Cython/Distutils/__pycache__/__init__.cpython-39.pyc,,
183
+ Cython/Distutils/__pycache__/build_ext.cpython-39.pyc,,
184
+ Cython/Distutils/__pycache__/extension.cpython-39.pyc,,
185
+ Cython/Distutils/__pycache__/old_build_ext.cpython-39.pyc,,
186
+ Cython/Distutils/build_ext.py,sha256=Fc_cI5wN0fT1Mf2k5B5nH-PgZ8Gq2lL6OlzF_qzy3dA,1007
187
+ Cython/Distutils/extension.py,sha256=FHvtK3Tj9MqE17TuZ_jWg1Mh4X7e-CXIPUpJK7nqcQE,4706
188
+ Cython/Distutils/old_build_ext.py,sha256=Hy34A1HqhoDOyU-krN2gJUYXK2mYWc8E2EZB-stvmrE,13635
189
+ Cython/Includes/Deprecated/python.pxd,sha256=l7crg8H9cVRedMcjDf_9xDLdnUT57Vt1BxlZWom-h88,61
190
+ Cython/Includes/Deprecated/python_bool.pxd,sha256=qOaFbsP6_pKoB3HGTjQUkFhQqukXheCmcSnnBQLdKGQ,66
191
+ Cython/Includes/Deprecated/python_buffer.pxd,sha256=gv2a3ngcOnRKZZHSox_bW1WD8jGbxfH9NJm1-iUXf9U,68
192
+ Cython/Includes/Deprecated/python_bytes.pxd,sha256=07-Hk3YpN_i4mIlbWYbNgDkjEytQAYOepJLJTY1CrVk,67
193
+ Cython/Includes/Deprecated/python_cobject.pxd,sha256=V9F0DHQbFZPbJ8RRnN9mft2ipq4wubM8ghBCGHr6NwE,69
194
+ Cython/Includes/Deprecated/python_complex.pxd,sha256=ITmq55v0b1gibEpLSCTCz68ViljenSuGGjiWn_nvIvI,69
195
+ Cython/Includes/Deprecated/python_dict.pxd,sha256=gYhGkJhMmzWcrXoPnJHUcp-vdtcwUACbGlfv3wtGsKU,66
196
+ Cython/Includes/Deprecated/python_exc.pxd,sha256=irWdwDYRWU16-P54uGDNfUSUtkL5Sj_1zBDWext_80g,65
197
+ Cython/Includes/Deprecated/python_float.pxd,sha256=v1Hbpd4SF3hSF7ZL_olMaYJzmBNA9jWn0eO9ggLBlvc,67
198
+ Cython/Includes/Deprecated/python_function.pxd,sha256=lkYKySQy1W36hfyyAJsc3E-8d9bsx5k8OhIMFQ6k2jA,70
199
+ Cython/Includes/Deprecated/python_getargs.pxd,sha256=NEdeqPqu4di0YJm_7yLfvuS903CAe4K2Pzb13TRfBdE,69
200
+ Cython/Includes/Deprecated/python_instance.pxd,sha256=FX9UlYrSxDrzch7wUvh_Y5Ix-bsDYARkXzZJOg2FvEI,70
201
+ Cython/Includes/Deprecated/python_int.pxd,sha256=Cwd4J4KTKjxwEMz1BbCso0g0pOID9AnySKOC1g0kLqA,65
202
+ Cython/Includes/Deprecated/python_iterator.pxd,sha256=nPJ0nKSmnUVzI1SPrTSt9wSD7SQILyhONJdP0H_-FGc,70
203
+ Cython/Includes/Deprecated/python_list.pxd,sha256=VHpylsg46-5Ud8rwlPe63bb3zSToXm9R_fPorZrJsUE,66
204
+ Cython/Includes/Deprecated/python_long.pxd,sha256=pg8hOKNoKaW-Mslugzeq6NCeznJw939LT24AVQn_cqE,66
205
+ Cython/Includes/Deprecated/python_mapping.pxd,sha256=AZtJdYm37glDSNChduAsgavz-_DPDkxxQEAO9lDGy84,69
206
+ Cython/Includes/Deprecated/python_mem.pxd,sha256=Mxidel5P4yuJxJOvoYr0PN1FD78oCOIJUEMPYMYU7lE,65
207
+ Cython/Includes/Deprecated/python_method.pxd,sha256=x5ye5_8KqtsW2HrEon5NdFJmIkmVDV1KeVpFsuC2UZE,68
208
+ Cython/Includes/Deprecated/python_module.pxd,sha256=lKu5VYCgC6S7LSgFa22V2YTY9JfML0vABDZpChhxs60,68
209
+ Cython/Includes/Deprecated/python_number.pxd,sha256=X4MxGoITZuJNPtC2cFJ8lQwui8MOC6rQfEDbFIcWA9k,68
210
+ Cython/Includes/Deprecated/python_object.pxd,sha256=qr2OwYVot4ELK3_-mCfaktXgLJEaKWDyCEblQ2vXV-E,68
211
+ Cython/Includes/Deprecated/python_oldbuffer.pxd,sha256=QyY4Vn5-cFaOt0oZ27GuRXa3tLawgMZN8KMamn9F1yo,71
212
+ Cython/Includes/Deprecated/python_pycapsule.pxd,sha256=tHJfhgm1TrSwJQwQFdhwP7YE7oQFiegxhNhgCDmlB6A,71
213
+ Cython/Includes/Deprecated/python_ref.pxd,sha256=wv39G35V7tN5sIhcL1APpe5NuhCwYwVy6X5DPPm5g5A,65
214
+ Cython/Includes/Deprecated/python_sequence.pxd,sha256=9ycCua1ODfECKPd56_GBmeqzWrfdqmkjhbEmdt87NC0,70
215
+ Cython/Includes/Deprecated/python_set.pxd,sha256=_Z5KVXs0V_T8fpgLX-2LbDAZIY1HnuhO-eTUHHRYwu0,65
216
+ Cython/Includes/Deprecated/python_string.pxd,sha256=6VgAehwW9PcUC9Kp_HbRVMYPeF_Q-L8yr9o2ezuTzys,68
217
+ Cython/Includes/Deprecated/python_tuple.pxd,sha256=_ZTQh7dRBmrRs9mtmOFjP37d0IFItxs20kzFtKtkY-g,67
218
+ Cython/Includes/Deprecated/python_type.pxd,sha256=2OKmEdSqoyK8fXttlHG3NRguZ-ZikUUet-kjKLq-eEU,66
219
+ Cython/Includes/Deprecated/python_unicode.pxd,sha256=TF8-N0un1WdyccTDo9hZVABc53SYzKnC3MEKrGb3vV0,69
220
+ Cython/Includes/Deprecated/python_version.pxd,sha256=ZXrK0UGUt8vHbYPxm7PTdhMe1_h7Yj6Lo74oFxjnNns,69
221
+ Cython/Includes/Deprecated/python_weakref.pxd,sha256=CUWMSmClrWPoTnlClOFCSHa6Xd55qDgIlcDCD6tfEhM,69
222
+ Cython/Includes/Deprecated/stdio.pxd,sha256=lNc2YuvWJ-LNSSdN7adDo1lf-C2M0r10hH4bysha9Sg,64
223
+ Cython/Includes/Deprecated/stdlib.pxd,sha256=PbCbjT8MjDjVRjx5Rod79gi22-9YI35jTulePAKCPXE,65
224
+ Cython/Includes/Deprecated/stl.pxd,sha256=tHpByeYgNiclr3YtCdKKAeEs3CHJflqacC7YgV7YN8k,2187
225
+ Cython/Includes/cpython/__init__.pxd,sha256=8URNRvb7JkYhqDZv2J0bVsdeZBEJBu7u2QFYkDyXPG8,8254
226
+ Cython/Includes/cpython/array.pxd,sha256=g6apBiXJG_7a0mjGqkFaqlcQjsg64uKK1VlXFFyXVCk,6056
227
+ Cython/Includes/cpython/bool.pxd,sha256=FaNn8K-Toq8FAws8BguKMk0IPM7IJm9IiUUGARSrKYk,1359
228
+ Cython/Includes/cpython/buffer.pxd,sha256=wm7aHygGUof_H3-JyICOek_xiU6Oks178ark1Nfk-a0,4870
229
+ Cython/Includes/cpython/bytearray.pxd,sha256=m0VdoHgouF1T0VtRjFLXZ5fi22vaMdVwFWpF3IxB6m4,1443
230
+ Cython/Includes/cpython/bytes.pxd,sha256=tGLuiBMzQjurK_pq77CM7P0C-Hn0KUIDZCXW9QvlJAI,9906
231
+ Cython/Includes/cpython/cellobject.pxd,sha256=DXdTjSN1RP1m4CsaGuggyIA1nGiIO4Kr7-c0ZWfrpRo,1390
232
+ Cython/Includes/cpython/ceval.pxd,sha256=h6fBetZCUvWTcCn3bkXZg2kqnIuyC5ZSChyhOocxVus,236
233
+ Cython/Includes/cpython/cobject.pxd,sha256=ZeMdbpZLqpcTywdv2VoppMTWD4X_yghL6Qox7LVfOyg,1524
234
+ Cython/Includes/cpython/codecs.pxd,sha256=3fyudEljkNGQ7e3dJPst6udXGcAeNKvlMK9U8EB1gXc,5084
235
+ Cython/Includes/cpython/complex.pxd,sha256=-bu0Cq91tS_U5tTra18S0jqt1FgSJTHXJ5J8rk-MOAA,1777
236
+ Cython/Includes/cpython/conversion.pxd,sha256=dbbFuZJF0SscmcaNCUf0tlBQDRdKYf5tH8yzhTU_XYI,1696
237
+ Cython/Includes/cpython/datetime.pxd,sha256=wQqB8i3tMZOTw9qrLdbHJRkxgZqscGEqmq0tIDfkkqw,6776
238
+ Cython/Includes/cpython/dict.pxd,sha256=F-mrlcAfNmTSUkpJed63bp1IaO0cwG56t_DLk7f0xv0,6877
239
+ Cython/Includes/cpython/exc.pxd,sha256=29-bGESwfoMqx1XU3MMggkIr8pz_l0UPruzy6KIzHxg,13606
240
+ Cython/Includes/cpython/float.pxd,sha256=RD1qEAUocXG9qXrRiT8aCSSfGEyTzjTc9HQkv5xg1ZE,1424
241
+ Cython/Includes/cpython/function.pxd,sha256=IoJUprbz8F10DEKh-vSSpY6nWkCHw7SqG9p2f-4gHek,2671
242
+ Cython/Includes/cpython/genobject.pxd,sha256=emC1JPgkuvBbGC0rgeZapKDaXYEj48uWiDC-xF0Mx2I,1052
243
+ Cython/Includes/cpython/getargs.pxd,sha256=268twKzdiAkQMXMsetNiNlNqaqzlKtiBENKbhOHd8x4,775
244
+ Cython/Includes/cpython/instance.pxd,sha256=qCbxPeHKOJbuszDu3UEaI-KLX9lTopuaNCcpoHJ9ngU,985
245
+ Cython/Includes/cpython/int.pxd,sha256=d9a0zUw_M3pRycCESWIjtfXWRvdvFOWxjdOjkcbX2gs,4131
246
+ Cython/Includes/cpython/iterator.pxd,sha256=o52mLHbdm14Kqant2hR2zAdYzqK4fkSWZtBcRmpoP-I,1319
247
+ Cython/Includes/cpython/iterobject.pxd,sha256=5UEZZwG5zyzxoCpknoQuh91zPUV11Uxr6F1taJdTv8k,1036
248
+ Cython/Includes/cpython/list.pxd,sha256=t-xo7ROcewe6-0ztrNjsxMKV2KxD-ILUzemQ2tTuI7E,4084
249
+ Cython/Includes/cpython/long.pxd,sha256=d6jHN1XJj7WL5PPAUK8U93IPyjWtlTmyhrBEVmxmGF8,7051
250
+ Cython/Includes/cpython/longintrepr.pxd,sha256=czvKr3fQdYIwIRu3gojXssT9LFXH-nstM7f_lPt7lE4,480
251
+ Cython/Includes/cpython/mapping.pxd,sha256=OIdvNVUoIpVCSQnkbLceTicSN0D_jRw6wQmbtxtxKuQ,2693
252
+ Cython/Includes/cpython/mem.pxd,sha256=AWVinanXFBZXvU141we2dD8dkOqMJ8W3KAAzpBJqB5g,5386
253
+ Cython/Includes/cpython/memoryview.pxd,sha256=l97J5-hbH3hp9aMbdXp3n73hJFNNsng6uyh40pc8P7I,2504
254
+ Cython/Includes/cpython/method.pxd,sha256=UWXflhIlP4y7B5XDbH9rQ15iADciGW-iqV1-dlw2Wwg,2196
255
+ Cython/Includes/cpython/module.pxd,sha256=Vc0Up7q1Mir38bN293E8RMugxWfuzjLFHM4g2dviPBM,9226
256
+ Cython/Includes/cpython/number.pxd,sha256=tYJ0nn0k_llUx3ilniW9iXd2rKVejA-J5UUiIJ36Kww,11922
257
+ Cython/Includes/cpython/object.pxd,sha256=AK5D-LrDbvisO6wpkh29G6xjA71sBF_KfKUyn0k2hzg,18366
258
+ Cython/Includes/cpython/oldbuffer.pxd,sha256=v0-YZ_Iwwj3ZQdM8VE5NPTQcbBlJdWwJGtNO9DonGgw,2916
259
+ Cython/Includes/cpython/pycapsule.pxd,sha256=8vySKea_zyTZZ1H39lICYkpJmnYTqZrun3DLf7d2294,5692
260
+ Cython/Includes/cpython/pylifecycle.pxd,sha256=LziJZHclGdtsr3yT28fULHNZ_n67bs1DmI9s8YzrBGg,2000
261
+ Cython/Includes/cpython/pystate.pxd,sha256=xgf1BBkv36qvqMaR77zZWYOuonAwe4RfNKE2g91A6fk,3683
262
+ Cython/Includes/cpython/pythread.pxd,sha256=0375TaYmtNCDDkWBh9WY4oJ_jhoTxhu_RR5QiOsXmYg,1946
263
+ Cython/Includes/cpython/ref.pxd,sha256=2AmgyGDhwA4scts0jcBTdGTCG0b2P8-eYAKFJk44x0I,2557
264
+ Cython/Includes/cpython/sequence.pxd,sha256=iTp3C6wOvTdvjLmdj3k9GqQqCGDlQFpzWi07wVQqSS4,6008
265
+ Cython/Includes/cpython/set.pxd,sha256=ewHRPVMbHUGDInZ3NziisCq68LvtmEJ-SXFbzmuJxLc,5383
266
+ Cython/Includes/cpython/slice.pxd,sha256=Rzgn8diAsN7lS2xGTq4VZucV3ziFNra4oz4tKGEAkMo,3111
267
+ Cython/Includes/cpython/string.pxd,sha256=EKjDGFnPcjnkndwGMJqRrszDV390Mc6o7AADChnNCiA,9944
268
+ Cython/Includes/cpython/tuple.pxd,sha256=eOLfH75ftJeYszztGFWWZP7LnyFOgw8GNuE7PQ9hAvs,3206
269
+ Cython/Includes/cpython/type.pxd,sha256=FOypwX0ZYamPc4uO8bejzO-HzgiaSRaXEPsxxxPIneI,1831
270
+ Cython/Includes/cpython/unicode.pxd,sha256=BYaF26EbiStNE62WHjLFFxAygoU9zwSezQDOpjxHse8,26242
271
+ Cython/Includes/cpython/version.pxd,sha256=l5KXt04isEv3qbGRJZ8fNlCYGO24HsA2l4EM3RxTEhE,847
272
+ Cython/Includes/cpython/weakref.pxd,sha256=UU9H_ovHG07FFgP_kY2xhGv3yJDr_8iujCZnxH2jnlo,1984
273
+ Cython/Includes/libc/__init__.pxd,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
274
+ Cython/Includes/libc/errno.pxd,sha256=j5hcKx7zinivU2b6SFMy8LZ9sJIQY5XLrp9cQUKv5AQ,2050
275
+ Cython/Includes/libc/float.pxd,sha256=IhvZJljpTG0fZtcIp7EBO2Sqddozxoxwj4RFNVcKLpY,966
276
+ Cython/Includes/libc/limits.pxd,sha256=xHlIyuDIKpjqclvRRYzZIcfd5G1re5QtbmoDMqZR_Ec,621
277
+ Cython/Includes/libc/locale.pxd,sha256=sixG8EJ6wiVb0HIR1LWJ3lXTjTv463GJ9C_40HRovN4,1140
278
+ Cython/Includes/libc/math.pxd,sha256=51YUxSe01R96_rr3sj4n4MLW-eOmQbcwdNn8YthTxqg,2948
279
+ Cython/Includes/libc/setjmp.pxd,sha256=XRh-gSuhvFLl0nRvz5OhSWYe9eqX2attAck3JI7mwa4,297
280
+ Cython/Includes/libc/signal.pxd,sha256=XOScPDA5vzlfEmu4D7DFT1-5Eu3qMpYdUarjt-fqlbw,1170
281
+ Cython/Includes/libc/stddef.pxd,sha256=0rCyoocCfDL-1OQo3pxHQ-6fW20SAYktOLPoa4d97w8,164
282
+ Cython/Includes/libc/stdint.pxd,sha256=qHJXzpWCrbvJWSaHYZL27VJPupQreTZl9VGj0jgLdRU,3449
283
+ Cython/Includes/libc/stdio.pxd,sha256=qUaxEwNrQl1-4yHLorzzJZ-a-y5_-Rm_m7Z5meaRqH0,2476
284
+ Cython/Includes/libc/stdlib.pxd,sha256=p62xq2XfB24WfNCjRXgD6cOYoRuV47AnYijkjWv4ugE,2444
285
+ Cython/Includes/libc/string.pxd,sha256=tzYGbRrnccedFLes-KGgJqM0FEtwHF_q4f2fqltNvyE,2038
286
+ Cython/Includes/libc/time.pxd,sha256=-IRH7fTq3wKBKmQQnpZRhaLsnl7D_qXFz_4BLB9O3u0,1317
287
+ Cython/Includes/libcpp/__init__.pxd,sha256=PCx8ZRfOeoyMRu41PPlPY9uo2kZmt_7d0KR4Epzfe7c,94
288
+ Cython/Includes/libcpp/algorithm.pxd,sha256=-2V0oR_cFbHHzeWT9RcfLvi5Oy-s_V2lO3OI6ZtX6fM,1770
289
+ Cython/Includes/libcpp/cast.pxd,sha256=En4LBubdinfpm9Rel077tK_LGwg_3k4FAu9mlIbKjuw,501
290
+ Cython/Includes/libcpp/complex.pxd,sha256=IjL8y9sAglhGbTKhqsJbW0mgMTYEUbYM1ekr5VDhQgY,3012
291
+ Cython/Includes/libcpp/deque.pxd,sha256=aWqZ9j3OgQuqFLkqRO_U2FIwbSe2fKmmYDRAfD0vGqU,3106
292
+ Cython/Includes/libcpp/forward_list.pxd,sha256=-So1ExEOkoPfsSdMlJSlI5665-zyWLMoUxlmm2Dlokk,2392
293
+ Cython/Includes/libcpp/functional.pxd,sha256=BXPYkffEOlKO1erTLqlkBLex6Gb5byDMF4hq_MZ2aVI,381
294
+ Cython/Includes/libcpp/iterator.pxd,sha256=mVc1rsAYfn_ARrdQ4JG-Ut5il5ynIa1CRXLk8Be8Zks,1432
295
+ Cython/Includes/libcpp/limits.pxd,sha256=RKV3wPvk4tV_vX5CYQRJIK5m5xXav7SeBxltlLyk8es,1661
296
+ Cython/Includes/libcpp/list.pxd,sha256=rGQfB3_mDcRkGKtMBuvDQvAPmgzR5jxSf3eOSRgR4YA,2658
297
+ Cython/Includes/libcpp/map.pxd,sha256=GF2sDnFBHZoU3Rcuo1rn6yKh45nhkX0_iH29xj581ow,2551
298
+ Cython/Includes/libcpp/memory.pxd,sha256=Hj20aSnmUTPAhFCrlmF_aeHJKiMiZ2bDKhaYn2yybJo,3600
299
+ Cython/Includes/libcpp/pair.pxd,sha256=UBJXw43uHkDlNsr0Pu1aP5tZ-ILXhUAyOLam2qdWmZA,27
300
+ Cython/Includes/libcpp/queue.pxd,sha256=FbL4Q7C3lgtZ2YzictU1XBXzQ7G-6y9i_7l2eqzA3Xc,649
301
+ Cython/Includes/libcpp/set.pxd,sha256=3y5Ir2TjGD7g3VRvlkXV1a3V3ZYzJvwOAfeTv8ucOCw,2170
302
+ Cython/Includes/libcpp/stack.pxd,sha256=zM3SQOqMWONVqud13ag3bUupA-ozU_YMq4Ad2QkL6fI,292
303
+ Cython/Includes/libcpp/string.pxd,sha256=zsvzyW6IggIDFqcF-UuLjxiNAHPtToAoc9VhKKG2U5A,8731
304
+ Cython/Includes/libcpp/typeindex.pxd,sha256=mIHr5Mq6Lol0SlzqeK6w_giVERh3uAjZm78yPDLXzc4,524
305
+ Cython/Includes/libcpp/typeinfo.pxd,sha256=tITsqurrdaZjsEGFksem9xZtVhSxQRxHZxcoC-4Y-DY,304
306
+ Cython/Includes/libcpp/unordered_map.pxd,sha256=eNna4hRAucQLnliBfEMu7Unfd_lB18I42iwKmsCO0-M,2867
307
+ Cython/Includes/libcpp/unordered_set.pxd,sha256=eUYSOMT5Gt8kZWCUKezQGyXWzatEyNg6-nmAlmcBo-k,2622
308
+ Cython/Includes/libcpp/utility.pxd,sha256=hTbvp7c12pnU2yvzzMvflZB-MAc_--3xh3PXtD_VIwg,1040
309
+ Cython/Includes/libcpp/vector.pxd,sha256=GYqLb74owhMmNQHUCcZSxGcYPgNuw6qULsfWKr7g6OQ,3350
310
+ Cython/Includes/numpy/__init__.pxd,sha256=CbLwvA4u-xj7RHxbO9Hs2o6hXd7GaJJlGEn9XJVH4c4,38138
311
+ Cython/Includes/numpy/math.pxd,sha256=qZEdamaPgCFW4J7Itc6BWgOrQSKZdxDT6kbU_gqx2g4,5807
312
+ Cython/Includes/openmp.pxd,sha256=orCIBYFuVPtLdRdhhCm5uhGbeV_fgVCA2Jk2Bts1e2g,1713
313
+ Cython/Includes/posix/__init__.pxd,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
314
+ Cython/Includes/posix/dlfcn.pxd,sha256=2IFcGBfZEmArdE0BxB71eT_Yb7n9STaVM11AtUcg_pE,355
315
+ Cython/Includes/posix/fcntl.pxd,sha256=oRL8-OsgcplHMGcYq5-W5twISvxK-vDfzAaEfuQHC-4,1194
316
+ Cython/Includes/posix/ioctl.pxd,sha256=2RC5zejPOCTkarDZM_6Vd2wc4oBuN7iaiL_C5MPBs90,99
317
+ Cython/Includes/posix/mman.pxd,sha256=juJcLi92N9Bc6L2p4zrUmYQIgNmrTsZ6hExbl1181pc,3362
318
+ Cython/Includes/posix/resource.pxd,sha256=MQe1bCTYQFVMsago3pgOvR6t6NElQElg7rhVANxYRcE,1254
319
+ Cython/Includes/posix/select.pxd,sha256=e4nhGHR8TRw6Xs9du5JoFtkd8U9sm3gX_BHq2FfmU6E,546
320
+ Cython/Includes/posix/signal.pxd,sha256=wFJI5UthdtU9mZWjEBeZ9IIfeX252JVwDk2tsbW_q3U,1876
321
+ Cython/Includes/posix/stat.pxd,sha256=ZOcPCpXnxlRRHcUkvg559hrFfB75uTbIYRWoQeyBCYs,1734
322
+ Cython/Includes/posix/stdio.pxd,sha256=K8DEH38hWMvy2A8zcKbHRrHSGsgwTIrQ9qCzU-0cWS0,1054
323
+ Cython/Includes/posix/stdlib.pxd,sha256=uGRPa00_HWZ6Chv5E13F96eut0xWHSfR7IioK9rKVLY,934
324
+ Cython/Includes/posix/strings.pxd,sha256=GNEteqND2wgXXSvkv6U9eKSC9oIom3C7o2zQ6W_J_S4,374
325
+ Cython/Includes/posix/time.pxd,sha256=wPUD7AjxpxmnUYmogTMFjroB2VzcPh8-b_8NEj-yG14,1980
326
+ Cython/Includes/posix/types.pxd,sha256=tWEWxST4EGHIgYS-Ce2SGjZ-KgmM2SVe1eggdcgv3JQ,1162
327
+ Cython/Includes/posix/unistd.pxd,sha256=w9B4d9NaXBsQ62XOr2xe9UFPGewmEk5BG6sqiRWdoM8,8061
328
+ Cython/Includes/posix/wait.pxd,sha256=WNogQvKu2hMfEQiCyaANfVWFnyJSk6TxBU0c6npeJrA,1244
329
+ Cython/Plex/Actions.cp39-win_amd64.pyd,sha256=kQc7WPOd47K3EAaUyB2-41BFr10WQETcWipUPWTPKH4,51712
330
+ Cython/Plex/Actions.pxd,sha256=FC-6ffzWR4i3rR6VSL2C64Xxs1qBhpBEzRsU7WpLn1Y,585
331
+ Cython/Plex/Actions.py,sha256=Caxkx8Kup9m4sx24ZcDTq-fAfPGG06TAHu2NI1D9zPs,2545
332
+ Cython/Plex/DFA.py,sha256=w4vl2ejXv6ptILtkTCbB8NcvF8ylwc6DaQ2gPFrWuo4,6012
333
+ Cython/Plex/Errors.py,sha256=As5uuGmqZe4w0B7Dm981lZTnDG-nlXSHYqiGUKnhrrY,1169
334
+ Cython/Plex/Lexicons.py,sha256=ay3yy9fqI5y5lfgpJ4ubBjYZQ53gFDVgNGbmoSl5DxI,6907
335
+ Cython/Plex/Machines.py,sha256=bIKg3-yxD_r7x-zEowJ7EsPBWlgXm_XhIozqsLQBeTk,7760
336
+ Cython/Plex/Regexps.py,sha256=qaP-Fr-GgKNmBVsMyXO3ltl2HH1JQcQiFmX2oyUyeOA,16208
337
+ Cython/Plex/Scanners.cp39-win_amd64.pyd,sha256=4_xLNnRnw6q1IjTxvfBqc63oEOzyCsgnnkpeqb0DROQ,87552
338
+ Cython/Plex/Scanners.pxd,sha256=oSfcDUZ3syc2ag73udwU5xoaIGDxiNd8a2F_LLw5PzY,1481
339
+ Cython/Plex/Scanners.py,sha256=-TXAxKW43ZbQNCSEkMWEJ0SmqYVVCkSOT9UngOCRZnQ,12259
340
+ Cython/Plex/Timing.py,sha256=-VgQveS-Ip_2ErjrVxh4w7cXpyVBkUaSaiLadyD3bw0,472
341
+ Cython/Plex/Traditional.py,sha256=cAT-pZnqIwCJaqgSqgKODSznFZ5DunUw_MLWx8Y650c,4120
342
+ Cython/Plex/Transitions.py,sha256=Tvp7cFXR3ZBPPHm0TAhUMC_-uiRR9YdOkF4t0wtk-f0,7187
343
+ Cython/Plex/__init__.py,sha256=dvMeQpSyZE75W0gkf4Xo5LAxgQLNhkAXiQoIOtcOkZ0,1282
344
+ Cython/Plex/__pycache__/Actions.cpython-39.pyc,,
345
+ Cython/Plex/__pycache__/DFA.cpython-39.pyc,,
346
+ Cython/Plex/__pycache__/Errors.cpython-39.pyc,,
347
+ Cython/Plex/__pycache__/Lexicons.cpython-39.pyc,,
348
+ Cython/Plex/__pycache__/Machines.cpython-39.pyc,,
349
+ Cython/Plex/__pycache__/Regexps.cpython-39.pyc,,
350
+ Cython/Plex/__pycache__/Scanners.cpython-39.pyc,,
351
+ Cython/Plex/__pycache__/Timing.cpython-39.pyc,,
352
+ Cython/Plex/__pycache__/Traditional.cpython-39.pyc,,
353
+ Cython/Plex/__pycache__/Transitions.cpython-39.pyc,,
354
+ Cython/Plex/__pycache__/__init__.cpython-39.pyc,,
355
+ Cython/Runtime/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
356
+ Cython/Runtime/__pycache__/__init__.cpython-39.pyc,,
357
+ Cython/Runtime/refnanny.cp39-win_amd64.pyd,sha256=_KIXM4YSRstJJfDb_l5sY-PYcjdASGIhrSIPYoeBO0U,63488
358
+ Cython/Runtime/refnanny.pyx,sha256=f2p1_0YxK25lm8Qfsu2ytvl0Im7GYyix1Q9krEBwC6c,6279
359
+ Cython/Shadow.py,sha256=h3hbgU48njWn6I7zGzC9plbwIn3GUSWaXu0a3g6dGbY,12987
360
+ Cython/StringIOTree.py,sha256=GX-TWn9XHwY5ecb4in8ovsTS5CtPTsSxZpanLWmQxgE,3336
361
+ Cython/Tempita/__init__.py,sha256=YHujYHiLoYUwFNNswJCgzSrDuie3sV08JsWT9Nbmp78,152
362
+ Cython/Tempita/__pycache__/__init__.cpython-39.pyc,,
363
+ Cython/Tempita/__pycache__/_looper.cpython-39.pyc,,
364
+ Cython/Tempita/__pycache__/_tempita.cpython-39.pyc,,
365
+ Cython/Tempita/__pycache__/compat3.cpython-39.pyc,,
366
+ Cython/Tempita/_looper.py,sha256=jlStYhz9Pgp6NatX86k-netBNBmvwaeWxCRS_S8vcIM,4168
367
+ Cython/Tempita/_tempita.cp39-win_amd64.pyd,sha256=mT6_MI8xN-4x6mcsuYTTSlveLNwdw9THuPZVrZu_u4U,383488
368
+ Cython/Tempita/_tempita.py,sha256=4gnJhuVIsGciu_5Besbvw26g82Pm7CiXazMghZO3ejs,39588
369
+ Cython/Tempita/compat3.py,sha256=cjW1y266vRF5Xvh8kAu7_qHGT8AGGu2kGSJRK6DI-0E,903
370
+ Cython/TestUtils.py,sha256=fzpic9xU-LP0wempXqwUQWZapBvXnFgbW_W9--IKpIA,7979
371
+ Cython/Tests/TestCodeWriter.py,sha256=qKad43J3hN7PLp7mVbEDESt96qsk8y3ELRwwIp9jnNw,2316
372
+ Cython/Tests/TestCythonUtils.py,sha256=XF4Fw4J5HZ4jUPLVv7ea8ZZcl2i9yXn5cx27WTtrcmU,474
373
+ Cython/Tests/TestJediTyper.py,sha256=F6MUG8SdzGXQwkbw6Wv1PqVlmlIT1z_7lH2buVOFT_I,6996
374
+ Cython/Tests/TestStringIOTree.py,sha256=vTuu3z32WTcmJaf0fBq62NMghYtaPL2rRnfdl2WM--4,1946
375
+ Cython/Tests/__init__.py,sha256=jOqtmPLCvMCq0xVMwGekuLpBmVgq0xtPFmUePySdOjs,13
376
+ Cython/Tests/__pycache__/TestCodeWriter.cpython-39.pyc,,
377
+ Cython/Tests/__pycache__/TestCythonUtils.cpython-39.pyc,,
378
+ Cython/Tests/__pycache__/TestJediTyper.cpython-39.pyc,,
379
+ Cython/Tests/__pycache__/TestStringIOTree.cpython-39.pyc,,
380
+ Cython/Tests/__pycache__/__init__.cpython-39.pyc,,
381
+ Cython/Tests/__pycache__/xmlrunner.cpython-39.pyc,,
382
+ Cython/Tests/xmlrunner.py,sha256=N1Z_C4Q_rSWFNQxm3L99qX-SaIWEksXbmfXOd_srg3s,14801
383
+ Cython/Utility/AsyncGen.c,sha256=iwNH8NdegHZYafEgBpErk6erU5o6GKVtHgqd3Vq9kNc,41023
384
+ Cython/Utility/Buffer.c,sha256=VUF4xHKJGX7QMTvpJO40aI1JUL-SERLEvlXXXEk2dHU,29654
385
+ Cython/Utility/Builtins.c,sha256=gYObNoiK_NVWRuzSFRePMb-dtw-XCp_Dx_Ztjmpq7as,16818
386
+ Cython/Utility/CConvert.pyx,sha256=fbZVHvm2vlWj2rgm8ajBt5jrbN30nY7dEmHlBCGomlU,4338
387
+ Cython/Utility/CMath.c,sha256=GIc7gd2WzaZryDJM3tefqXifLJpUJs6_T_c_mFrr-s8,2566
388
+ Cython/Utility/Capsule.c,sha256=SOeU7E7T7piQEx894T2QFH2RlSG-MmsiyuY4lVN1yso,505
389
+ Cython/Utility/CommonStructures.c,sha256=p65HHgTrf7h7Tj7JK7tIgkLrrCrjouL8HL90EHfoMoU,2558
390
+ Cython/Utility/Complex.c,sha256=J4HseVcBOzGo5dye0Gus8bf8rGwWLEzN9sRJM74SWiI,10043
391
+ Cython/Utility/Coroutine.c,sha256=v6efWVaz_2AtK9DWzqVmWMsQNR9k8pSOwZv3DXrVX9w,89431
392
+ Cython/Utility/CpdefEnums.pyx,sha256=XMg8sdltQSNj2wGVfnHIWRvyHFCcLK8ZfpKznKi4lhY,1893
393
+ Cython/Utility/CppConvert.pyx,sha256=-e5i3_J1SS_GbctsflQwylx9cqdk_CJ2SfQSEDHa71k,6098
394
+ Cython/Utility/CppSupport.cpp,sha256=NTnSRCmi2PHuT3J6Qy15xMZGx0bf9l-MaxAbW6OVk6s,2234
395
+ Cython/Utility/CythonFunction.c,sha256=G4AZwEnFrshqgFOxaE62zRW4pGcMC3zVzPoX6Ce8jR0,45993
396
+ Cython/Utility/Embed.c,sha256=sMDv7XVJswaGRTQbQHtEDThZaAmvbn-6yeIqrUzboL4,6854
397
+ Cython/Utility/Exceptions.c,sha256=AsCDXV4cNUC4jzCe_xYdqRFNiUkgkQXjZXZ4LOo9KBY,26719
398
+ Cython/Utility/ExtensionTypes.c,sha256=MugMfJZ99S2aNJ3uX4gPuqxTwfMOdiWkSNJnRGd0bIM,10899
399
+ Cython/Utility/FunctionArguments.c,sha256=IH9Y5aV_tNrJLo_CWHskEnFai9fp9cKLvRkIZYl2UGQ,12040
400
+ Cython/Utility/ImportExport.c,sha256=KE8VQNH8IXLjquRCVxw4mKV2ZwbAw2dSk7I2uTh0fLs,22287
401
+ Cython/Utility/MemoryView.pyx,sha256=vCbGT3WDoaB9gxv4eUr4UGS8YhHy3LK3E_r329s-L_k,49621
402
+ Cython/Utility/MemoryView_C.c,sha256=oz4b-wJSQSxS8e6QFG0zeNJxCognXUohM-FS-RbFBRs,29243
403
+ Cython/Utility/ModuleSetupCode.c,sha256=8iQSelhOKD_x_IpX847nGE3I_20itjlEvxQUEaZTnXA,55643
404
+ Cython/Utility/ObjectHandling.c,sha256=RhHh3OF0ahDiwvtwSA6kIRCeJcE7-C49bJurqSdkgwU,88626
405
+ Cython/Utility/Optimize.c,sha256=xtKZ8WhOe8l4UYONGVDilNFhzGMueJxA7nYJn8QIWmc,45184
406
+ Cython/Utility/Overflow.c,sha256=_KXlJsbMIi-jzdCotwxkN6mtqo6jHRNnPJ1ZKwXVhpE,12424
407
+ Cython/Utility/Printing.c,sha256=o8XnfjNIT8Ub5KY4FAp_FNw-OE3xqjy0MgmYWgDcWao,5103
408
+ Cython/Utility/Profile.c,sha256=Vq5yCvDISKhLGVwTxQHIQ5nL5aYRfYpg8FijoDb2cBc,17784
409
+ Cython/Utility/StringTools.c,sha256=a9xeHRIKkIkk89joQCTkMR4ibJM20PZDuuDEf2yQgXU,42181
410
+ Cython/Utility/TestCyUtilityLoader.pyx,sha256=91lWWJub7l_6xNn3ncrvQZZ94RpkQzEx2NtAaFpvrxY,152
411
+ Cython/Utility/TestCythonScope.pyx,sha256=HQm5E5Eehr3tkDDURURyVnDputKG3-Wn2k2aIAoru9g,1595
412
+ Cython/Utility/TestUtilityLoader.c,sha256=dGy6ZWL2kBqtmUY7kF75UEox5kadQZ__BmZKscwg2aY,279
413
+ Cython/Utility/TypeConversion.c,sha256=0K3erVzNT9lY-jEygTxsarAirETGZcOzvSK-VDs5EJY,36302
414
+ Cython/Utility/__init__.py,sha256=t2bpY-TYSX8lJdbKuBFJ1kBfpWVzgGw4xoZlCKfyj_s,1159
415
+ Cython/Utility/__pycache__/__init__.cpython-39.pyc,,
416
+ Cython/Utility/arrayarray.h,sha256=3Ll8Gd_S4rv8HaTfg5i6-aaoB9taI1vzwTp7NeA7Wy0,4089
417
+ Cython/Utils.py,sha256=YJa4RP6fSQkWpbA_X0DnNREd3Y5NuPDjGwUdNvVW2OM,13458
418
+ Cython/__init__.py,sha256=GMnkoIas6hfN_meqZAJF9BEs1NuY4-4B2L0Uls7hXaA,358
419
+ Cython/__pycache__/CodeWriter.cpython-39.pyc,,
420
+ Cython/__pycache__/Coverage.cpython-39.pyc,,
421
+ Cython/__pycache__/Debugging.cpython-39.pyc,,
422
+ Cython/__pycache__/Shadow.cpython-39.pyc,,
423
+ Cython/__pycache__/StringIOTree.cpython-39.pyc,,
424
+ Cython/__pycache__/TestUtils.cpython-39.pyc,,
425
+ Cython/__pycache__/Utils.cpython-39.pyc,,
426
+ Cython/__pycache__/__init__.cpython-39.pyc,,
427
+ __pycache__/cython.cpython-39.pyc,,
428
+ cython.py,sha256=z2AtgHBGh0x0h0ZcGje7IhYlR6nGH_MmOh1fFMjqYn0,520
429
+ pyximport/__init__.py,sha256=9hOyKolFtOerPiVEyktKrT1VtzbGexq9UmORzo52iHI,79
430
+ pyximport/__pycache__/__init__.cpython-39.pyc,,
431
+ pyximport/__pycache__/pyxbuild.cpython-39.pyc,,
432
+ pyximport/__pycache__/pyximport.cpython-39.pyc,,
433
+ pyximport/pyxbuild.py,sha256=TiAkhtSxSbRW04JKtgO3FP3hfVzQ1mjjzCh5PqZDOrM,5702
434
+ pyximport/pyximport.py,sha256=Vjxp3kbmFRf9j0ya4f0m0Ahytkjjmv2UkFueasXxL5A,23578
Cython-0.29.28.dist-info/REQUESTED ADDED
File without changes
Cython-0.29.28.dist-info/WHEEL ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ Wheel-Version: 1.0
2
+ Generator: bdist_wheel (0.37.1)
3
+ Root-Is-Purelib: false
4
+ Tag: cp39-cp39-win_amd64
5
+
Cython-0.29.28.dist-info/direct_url.json ADDED
@@ -0,0 +1 @@
 
 
1
+ {"dir_info": {}, "url": "file:///C:/ci/cython_1647850559892/work"}
Cython-0.29.28.dist-info/entry_points.txt ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ [console_scripts]
2
+ cygdb = Cython.Debugger.Cygdb:main
3
+ cython = Cython.Compiler.Main:setuptools_main
4
+ cythonize = Cython.Build.Cythonize:main
5
+
Cython-0.29.28.dist-info/top_level.txt ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ Cython
2
+ cython
3
+ pyximport
Cython/Build/BuildExecutable.py ADDED
@@ -0,0 +1,142 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """
2
+ Compile a Python script into an executable that embeds CPython and run it.
3
+ Requires CPython to be built as a shared library ('libpythonX.Y').
4
+
5
+ Basic usage:
6
+
7
+ python cythonrun somefile.py [ARGS]
8
+ """
9
+
10
+ from __future__ import absolute_import
11
+
12
+ DEBUG = True
13
+
14
+ import sys
15
+ import os
16
+ from distutils import sysconfig
17
+
18
+
19
+ def get_config_var(name, default=''):
20
+ return sysconfig.get_config_var(name) or default
21
+
22
+ INCDIR = sysconfig.get_python_inc()
23
+ LIBDIR1 = get_config_var('LIBDIR')
24
+ LIBDIR2 = get_config_var('LIBPL')
25
+ PYLIB = get_config_var('LIBRARY')
26
+ PYLIB_DYN = get_config_var('LDLIBRARY')
27
+ if PYLIB_DYN == PYLIB:
28
+ # no shared library
29
+ PYLIB_DYN = ''
30
+ else:
31
+ PYLIB_DYN = os.path.splitext(PYLIB_DYN[3:])[0] # 'lib(XYZ).so' -> XYZ
32
+
33
+ CC = get_config_var('CC', os.environ.get('CC', ''))
34
+ CFLAGS = get_config_var('CFLAGS') + ' ' + os.environ.get('CFLAGS', '')
35
+ LINKCC = get_config_var('LINKCC', os.environ.get('LINKCC', CC))
36
+ LINKFORSHARED = get_config_var('LINKFORSHARED')
37
+ LIBS = get_config_var('LIBS')
38
+ SYSLIBS = get_config_var('SYSLIBS')
39
+ EXE_EXT = sysconfig.get_config_var('EXE')
40
+
41
+ def _debug(msg, *args):
42
+ if DEBUG:
43
+ if args:
44
+ msg = msg % args
45
+ sys.stderr.write(msg + '\n')
46
+
47
+ def dump_config():
48
+ _debug('INCDIR: %s', INCDIR)
49
+ _debug('LIBDIR1: %s', LIBDIR1)
50
+ _debug('LIBDIR2: %s', LIBDIR2)
51
+ _debug('PYLIB: %s', PYLIB)
52
+ _debug('PYLIB_DYN: %s', PYLIB_DYN)
53
+ _debug('CC: %s', CC)
54
+ _debug('CFLAGS: %s', CFLAGS)
55
+ _debug('LINKCC: %s', LINKCC)
56
+ _debug('LINKFORSHARED: %s', LINKFORSHARED)
57
+ _debug('LIBS: %s', LIBS)
58
+ _debug('SYSLIBS: %s', SYSLIBS)
59
+ _debug('EXE_EXT: %s', EXE_EXT)
60
+
61
+ def runcmd(cmd, shell=True):
62
+ if shell:
63
+ cmd = ' '.join(cmd)
64
+ _debug(cmd)
65
+ else:
66
+ _debug(' '.join(cmd))
67
+
68
+ try:
69
+ import subprocess
70
+ except ImportError: # Python 2.3 ...
71
+ returncode = os.system(cmd)
72
+ else:
73
+ returncode = subprocess.call(cmd, shell=shell)
74
+
75
+ if returncode:
76
+ sys.exit(returncode)
77
+
78
+ def clink(basename):
79
+ runcmd([LINKCC, '-o', basename + EXE_EXT, basename+'.o', '-L'+LIBDIR1, '-L'+LIBDIR2]
80
+ + [PYLIB_DYN and ('-l'+PYLIB_DYN) or os.path.join(LIBDIR1, PYLIB)]
81
+ + LIBS.split() + SYSLIBS.split() + LINKFORSHARED.split())
82
+
83
+ def ccompile(basename):
84
+ runcmd([CC, '-c', '-o', basename+'.o', basename+'.c', '-I' + INCDIR] + CFLAGS.split())
85
+
86
+ def cycompile(input_file, options=()):
87
+ from ..Compiler import Version, CmdLine, Main
88
+ options, sources = CmdLine.parse_command_line(list(options or ()) + ['--embed', input_file])
89
+ _debug('Using Cython %s to compile %s', Version.version, input_file)
90
+ result = Main.compile(sources, options)
91
+ if result.num_errors > 0:
92
+ sys.exit(1)
93
+
94
+ def exec_file(program_name, args=()):
95
+ runcmd([os.path.abspath(program_name)] + list(args), shell=False)
96
+
97
+ def build(input_file, compiler_args=(), force=False):
98
+ """
99
+ Build an executable program from a Cython module.
100
+
101
+ Returns the name of the executable file.
102
+ """
103
+ basename = os.path.splitext(input_file)[0]
104
+ exe_file = basename + EXE_EXT
105
+ if not force and os.path.abspath(exe_file) == os.path.abspath(input_file):
106
+ raise ValueError("Input and output file names are the same, refusing to overwrite")
107
+ if (not force and os.path.exists(exe_file) and os.path.exists(input_file)
108
+ and os.path.getmtime(input_file) <= os.path.getmtime(exe_file)):
109
+ _debug("File is up to date, not regenerating %s", exe_file)
110
+ return exe_file
111
+ cycompile(input_file, compiler_args)
112
+ ccompile(basename)
113
+ clink(basename)
114
+ return exe_file
115
+
116
+ def build_and_run(args):
117
+ """
118
+ Build an executable program from a Cython module and runs it.
119
+
120
+ Arguments after the module name will be passed verbatimely to the
121
+ program.
122
+ """
123
+ cy_args = []
124
+ last_arg = None
125
+ for i, arg in enumerate(args):
126
+ if arg.startswith('-'):
127
+ cy_args.append(arg)
128
+ elif last_arg in ('-X', '--directive'):
129
+ cy_args.append(arg)
130
+ else:
131
+ input_file = arg
132
+ args = args[i+1:]
133
+ break
134
+ last_arg = arg
135
+ else:
136
+ raise ValueError('no input file provided')
137
+
138
+ program_name = build(input_file, cy_args)
139
+ exec_file(program_name, args)
140
+
141
+ if __name__ == '__main__':
142
+ build_and_run(sys.argv[1:])
Cython/Build/Cythonize.py ADDED
@@ -0,0 +1,229 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #!/usr/bin/env python
2
+
3
+ from __future__ import absolute_import
4
+
5
+ import os
6
+ import shutil
7
+ import tempfile
8
+ from distutils.core import setup
9
+
10
+ from .Dependencies import cythonize, extended_iglob
11
+ from ..Utils import is_package_dir
12
+ from ..Compiler import Options
13
+
14
+ try:
15
+ import multiprocessing
16
+ parallel_compiles = int(multiprocessing.cpu_count() * 1.5)
17
+ except ImportError:
18
+ multiprocessing = None
19
+ parallel_compiles = 0
20
+
21
+
22
+ class _FakePool(object):
23
+ def map_async(self, func, args):
24
+ try:
25
+ from itertools import imap
26
+ except ImportError:
27
+ imap=map
28
+ for _ in imap(func, args):
29
+ pass
30
+
31
+ def close(self):
32
+ pass
33
+
34
+ def terminate(self):
35
+ pass
36
+
37
+ def join(self):
38
+ pass
39
+
40
+
41
+ def parse_directives(option, name, value, parser):
42
+ dest = option.dest
43
+ old_directives = dict(getattr(parser.values, dest,
44
+ Options.get_directive_defaults()))
45
+ directives = Options.parse_directive_list(
46
+ value, relaxed_bool=True, current_settings=old_directives)
47
+ setattr(parser.values, dest, directives)
48
+
49
+
50
+ def parse_options(option, name, value, parser):
51
+ dest = option.dest
52
+ options = dict(getattr(parser.values, dest, {}))
53
+ for opt in value.split(','):
54
+ if '=' in opt:
55
+ n, v = opt.split('=', 1)
56
+ v = v.lower() not in ('false', 'f', '0', 'no')
57
+ else:
58
+ n, v = opt, True
59
+ options[n] = v
60
+ setattr(parser.values, dest, options)
61
+
62
+
63
+ def parse_compile_time_env(option, name, value, parser):
64
+ dest = option.dest
65
+ old_env = dict(getattr(parser.values, dest, {}))
66
+ new_env = Options.parse_compile_time_env(value, current_settings=old_env)
67
+ setattr(parser.values, dest, new_env)
68
+
69
+
70
+ def find_package_base(path):
71
+ base_dir, package_path = os.path.split(path)
72
+ while os.path.isfile(os.path.join(base_dir, '__init__.py')):
73
+ base_dir, parent = os.path.split(base_dir)
74
+ package_path = '%s/%s' % (parent, package_path)
75
+ return base_dir, package_path
76
+
77
+
78
+ def cython_compile(path_pattern, options):
79
+ pool = None
80
+ all_paths = map(os.path.abspath, extended_iglob(path_pattern))
81
+ try:
82
+ for path in all_paths:
83
+ if options.build_inplace:
84
+ base_dir = path
85
+ while not os.path.isdir(base_dir) or is_package_dir(base_dir):
86
+ base_dir = os.path.dirname(base_dir)
87
+ else:
88
+ base_dir = None
89
+
90
+ if os.path.isdir(path):
91
+ # recursively compiling a package
92
+ paths = [os.path.join(path, '**', '*.{py,pyx}')]
93
+ else:
94
+ # assume it's a file(-like thing)
95
+ paths = [path]
96
+
97
+ ext_modules = cythonize(
98
+ paths,
99
+ nthreads=options.parallel,
100
+ exclude_failures=options.keep_going,
101
+ exclude=options.excludes,
102
+ compiler_directives=options.directives,
103
+ compile_time_env=options.compile_time_env,
104
+ force=options.force,
105
+ quiet=options.quiet,
106
+ depfile=options.depfile,
107
+ **options.options)
108
+
109
+ if ext_modules and options.build:
110
+ if len(ext_modules) > 1 and options.parallel > 1:
111
+ if pool is None:
112
+ try:
113
+ pool = multiprocessing.Pool(options.parallel)
114
+ except OSError:
115
+ pool = _FakePool()
116
+ pool.map_async(run_distutils, [
117
+ (base_dir, [ext]) for ext in ext_modules])
118
+ else:
119
+ run_distutils((base_dir, ext_modules))
120
+ except:
121
+ if pool is not None:
122
+ pool.terminate()
123
+ raise
124
+ else:
125
+ if pool is not None:
126
+ pool.close()
127
+ pool.join()
128
+
129
+
130
+ def run_distutils(args):
131
+ base_dir, ext_modules = args
132
+ script_args = ['build_ext', '-i']
133
+ cwd = os.getcwd()
134
+ temp_dir = None
135
+ try:
136
+ if base_dir:
137
+ os.chdir(base_dir)
138
+ temp_dir = tempfile.mkdtemp(dir=base_dir)
139
+ script_args.extend(['--build-temp', temp_dir])
140
+ setup(
141
+ script_name='setup.py',
142
+ script_args=script_args,
143
+ ext_modules=ext_modules,
144
+ )
145
+ finally:
146
+ if base_dir:
147
+ os.chdir(cwd)
148
+ if temp_dir and os.path.isdir(temp_dir):
149
+ shutil.rmtree(temp_dir)
150
+
151
+
152
+ def parse_args(args):
153
+ from optparse import OptionParser
154
+ parser = OptionParser(usage='%prog [options] [sources and packages]+')
155
+
156
+ parser.add_option('-X', '--directive', metavar='NAME=VALUE,...',
157
+ dest='directives', default={}, type="str",
158
+ action='callback', callback=parse_directives,
159
+ help='set a compiler directive')
160
+ parser.add_option('-E', '--compile-time-env', metavar='NAME=VALUE,...',
161
+ dest='compile_time_env', default={}, type="str",
162
+ action='callback', callback=parse_compile_time_env,
163
+ help='set a compile time environment variable')
164
+ parser.add_option('-s', '--option', metavar='NAME=VALUE',
165
+ dest='options', default={}, type="str",
166
+ action='callback', callback=parse_options,
167
+ help='set a cythonize option')
168
+ parser.add_option('-2', dest='language_level', action='store_const', const=2, default=None,
169
+ help='use Python 2 syntax mode by default')
170
+ parser.add_option('-3', dest='language_level', action='store_const', const=3,
171
+ help='use Python 3 syntax mode by default')
172
+ parser.add_option('--3str', dest='language_level', action='store_const', const='3str',
173
+ help='use Python 3 syntax mode by default')
174
+ parser.add_option('-a', '--annotate', dest='annotate', action='store_true',
175
+ help='generate annotated HTML page for source files')
176
+
177
+ parser.add_option('-x', '--exclude', metavar='PATTERN', dest='excludes',
178
+ action='append', default=[],
179
+ help='exclude certain file patterns from the compilation')
180
+
181
+ parser.add_option('-b', '--build', dest='build', action='store_true',
182
+ help='build extension modules using distutils')
183
+ parser.add_option('-i', '--inplace', dest='build_inplace', action='store_true',
184
+ help='build extension modules in place using distutils (implies -b)')
185
+ parser.add_option('-j', '--parallel', dest='parallel', metavar='N',
186
+ type=int, default=parallel_compiles,
187
+ help=('run builds in N parallel jobs (default: %d)' %
188
+ parallel_compiles or 1))
189
+ parser.add_option('-f', '--force', dest='force', action='store_true',
190
+ help='force recompilation')
191
+ parser.add_option('-q', '--quiet', dest='quiet', action='store_true',
192
+ help='be less verbose during compilation')
193
+
194
+ parser.add_option('--lenient', dest='lenient', action='store_true',
195
+ help='increase Python compatibility by ignoring some compile time errors')
196
+ parser.add_option('-k', '--keep-going', dest='keep_going', action='store_true',
197
+ help='compile as much as possible, ignore compilation failures')
198
+ parser.add_option('-M', '--depfile', action='store_true', help='produce depfiles for the sources')
199
+
200
+ options, args = parser.parse_args(args)
201
+ if not args:
202
+ parser.error("no source files provided")
203
+ if options.build_inplace:
204
+ options.build = True
205
+ if multiprocessing is None:
206
+ options.parallel = 0
207
+ if options.language_level:
208
+ assert options.language_level in (2, 3, '3str')
209
+ options.options['language_level'] = options.language_level
210
+ return options, args
211
+
212
+
213
+ def main(args=None):
214
+ options, paths = parse_args(args)
215
+
216
+ if options.lenient:
217
+ # increase Python compatibility by ignoring compile time errors
218
+ Options.error_on_unknown_names = False
219
+ Options.error_on_uninitialized = False
220
+
221
+ if options.annotate:
222
+ Options.annotate = True
223
+
224
+ for path in paths:
225
+ cython_compile(path, options)
226
+
227
+
228
+ if __name__ == '__main__':
229
+ main()
Cython/Build/Dependencies.py ADDED
@@ -0,0 +1,1308 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import, print_function
2
+
3
+ import cython
4
+ from .. import __version__
5
+
6
+ import collections
7
+ import contextlib
8
+ import hashlib
9
+ import os
10
+ import shutil
11
+ import subprocess
12
+ import re, sys, time
13
+ import warnings
14
+ from glob import iglob
15
+ from io import open as io_open
16
+ from os.path import relpath as _relpath
17
+ from distutils.extension import Extension
18
+ from distutils.util import strtobool
19
+ import zipfile
20
+
21
+ try:
22
+ from collections.abc import Iterable
23
+ except ImportError:
24
+ from collections import Iterable
25
+
26
+ try:
27
+ import gzip
28
+ gzip_open = gzip.open
29
+ gzip_ext = '.gz'
30
+ except ImportError:
31
+ gzip_open = open
32
+ gzip_ext = ''
33
+
34
+ try:
35
+ import zlib
36
+ zipfile_compression_mode = zipfile.ZIP_DEFLATED
37
+ except ImportError:
38
+ zipfile_compression_mode = zipfile.ZIP_STORED
39
+
40
+ try:
41
+ import pythran
42
+ except:
43
+ pythran = None
44
+
45
+ from .. import Utils
46
+ from ..Utils import (cached_function, cached_method, path_exists,
47
+ safe_makedirs, copy_file_to_dir_if_newer, is_package_dir, replace_suffix)
48
+ from ..Compiler.Main import Context, CompilationOptions, default_options
49
+
50
+ join_path = cached_function(os.path.join)
51
+ copy_once_if_newer = cached_function(copy_file_to_dir_if_newer)
52
+ safe_makedirs_once = cached_function(safe_makedirs)
53
+
54
+ if sys.version_info[0] < 3:
55
+ # stupid Py2 distutils enforces str type in list of sources
56
+ _fs_encoding = sys.getfilesystemencoding()
57
+ if _fs_encoding is None:
58
+ _fs_encoding = sys.getdefaultencoding()
59
+ def encode_filename_in_py2(filename):
60
+ if not isinstance(filename, bytes):
61
+ return filename.encode(_fs_encoding)
62
+ return filename
63
+ else:
64
+ def encode_filename_in_py2(filename):
65
+ return filename
66
+ basestring = str
67
+
68
+
69
+ def _make_relative(file_paths, base=None):
70
+ if not base:
71
+ base = os.getcwd()
72
+ if base[-1] != os.path.sep:
73
+ base += os.path.sep
74
+ return [_relpath(path, base) if path.startswith(base) else path
75
+ for path in file_paths]
76
+
77
+
78
+ def extended_iglob(pattern):
79
+ if '{' in pattern:
80
+ m = re.match('(.*){([^}]+)}(.*)', pattern)
81
+ if m:
82
+ before, switch, after = m.groups()
83
+ for case in switch.split(','):
84
+ for path in extended_iglob(before + case + after):
85
+ yield path
86
+ return
87
+ if '**/' in pattern:
88
+ seen = set()
89
+ first, rest = pattern.split('**/', 1)
90
+ if first:
91
+ first = iglob(first+'/')
92
+ else:
93
+ first = ['']
94
+ for root in first:
95
+ for path in extended_iglob(join_path(root, rest)):
96
+ if path not in seen:
97
+ seen.add(path)
98
+ yield path
99
+ for path in extended_iglob(join_path(root, '*', '**/' + rest)):
100
+ if path not in seen:
101
+ seen.add(path)
102
+ yield path
103
+ else:
104
+ for path in iglob(pattern):
105
+ yield path
106
+
107
+
108
+ def nonempty(it, error_msg="expected non-empty iterator"):
109
+ empty = True
110
+ for value in it:
111
+ empty = False
112
+ yield value
113
+ if empty:
114
+ raise ValueError(error_msg)
115
+
116
+
117
+ @cached_function
118
+ def file_hash(filename):
119
+ path = os.path.normpath(filename)
120
+ prefix = ('%d:%s' % (len(path), path)).encode("UTF-8")
121
+ m = hashlib.md5(prefix)
122
+ with open(path, 'rb') as f:
123
+ data = f.read(65000)
124
+ while data:
125
+ m.update(data)
126
+ data = f.read(65000)
127
+ return m.hexdigest()
128
+
129
+
130
+ def update_pythran_extension(ext):
131
+ if pythran is None:
132
+ raise RuntimeError("You first need to install Pythran to use the np_pythran directive.")
133
+ try:
134
+ pythran_ext = pythran.config.make_extension(python=True)
135
+ except TypeError: # older pythran version only
136
+ pythran_ext = pythran.config.make_extension()
137
+
138
+ ext.include_dirs.extend(pythran_ext['include_dirs'])
139
+ ext.extra_compile_args.extend(pythran_ext['extra_compile_args'])
140
+ ext.extra_link_args.extend(pythran_ext['extra_link_args'])
141
+ ext.define_macros.extend(pythran_ext['define_macros'])
142
+ ext.undef_macros.extend(pythran_ext['undef_macros'])
143
+ ext.library_dirs.extend(pythran_ext['library_dirs'])
144
+ ext.libraries.extend(pythran_ext['libraries'])
145
+ ext.language = 'c++'
146
+
147
+ # These options are not compatible with the way normal Cython extensions work
148
+ for bad_option in ["-fwhole-program", "-fvisibility=hidden"]:
149
+ try:
150
+ ext.extra_compile_args.remove(bad_option)
151
+ except ValueError:
152
+ pass
153
+
154
+
155
+ def parse_list(s):
156
+ """
157
+ >>> parse_list("")
158
+ []
159
+ >>> parse_list("a")
160
+ ['a']
161
+ >>> parse_list("a b c")
162
+ ['a', 'b', 'c']
163
+ >>> parse_list("[a, b, c]")
164
+ ['a', 'b', 'c']
165
+ >>> parse_list('a " " b')
166
+ ['a', ' ', 'b']
167
+ >>> parse_list('[a, ",a", "a,", ",", ]')
168
+ ['a', ',a', 'a,', ',']
169
+ """
170
+ if len(s) >= 2 and s[0] == '[' and s[-1] == ']':
171
+ s = s[1:-1]
172
+ delimiter = ','
173
+ else:
174
+ delimiter = ' '
175
+ s, literals = strip_string_literals(s)
176
+ def unquote(literal):
177
+ literal = literal.strip()
178
+ if literal[0] in "'\"":
179
+ return literals[literal[1:-1]]
180
+ else:
181
+ return literal
182
+ return [unquote(item) for item in s.split(delimiter) if item.strip()]
183
+
184
+
185
+ transitive_str = object()
186
+ transitive_list = object()
187
+ bool_or = object()
188
+
189
+ distutils_settings = {
190
+ 'name': str,
191
+ 'sources': list,
192
+ 'define_macros': list,
193
+ 'undef_macros': list,
194
+ 'libraries': transitive_list,
195
+ 'library_dirs': transitive_list,
196
+ 'runtime_library_dirs': transitive_list,
197
+ 'include_dirs': transitive_list,
198
+ 'extra_objects': list,
199
+ 'extra_compile_args': transitive_list,
200
+ 'extra_link_args': transitive_list,
201
+ 'export_symbols': list,
202
+ 'depends': transitive_list,
203
+ 'language': transitive_str,
204
+ 'np_pythran': bool_or
205
+ }
206
+
207
+
208
+ @cython.locals(start=cython.Py_ssize_t, end=cython.Py_ssize_t)
209
+ def line_iter(source):
210
+ if isinstance(source, basestring):
211
+ start = 0
212
+ while True:
213
+ end = source.find('\n', start)
214
+ if end == -1:
215
+ yield source[start:]
216
+ return
217
+ yield source[start:end]
218
+ start = end+1
219
+ else:
220
+ for line in source:
221
+ yield line
222
+
223
+
224
+ class DistutilsInfo(object):
225
+
226
+ def __init__(self, source=None, exn=None):
227
+ self.values = {}
228
+ if source is not None:
229
+ for line in line_iter(source):
230
+ line = line.lstrip()
231
+ if not line:
232
+ continue
233
+ if line[0] != '#':
234
+ break
235
+ line = line[1:].lstrip()
236
+ kind = next((k for k in ("distutils:","cython:") if line.startswith(k)), None)
237
+ if kind is not None:
238
+ key, _, value = [s.strip() for s in line[len(kind):].partition('=')]
239
+ type = distutils_settings.get(key, None)
240
+ if line.startswith("cython:") and type is None: continue
241
+ if type in (list, transitive_list):
242
+ value = parse_list(value)
243
+ if key == 'define_macros':
244
+ value = [tuple(macro.split('=', 1))
245
+ if '=' in macro else (macro, None)
246
+ for macro in value]
247
+ if type is bool_or:
248
+ value = strtobool(value)
249
+ self.values[key] = value
250
+ elif exn is not None:
251
+ for key in distutils_settings:
252
+ if key in ('name', 'sources','np_pythran'):
253
+ continue
254
+ value = getattr(exn, key, None)
255
+ if value:
256
+ self.values[key] = value
257
+
258
+ def merge(self, other):
259
+ if other is None:
260
+ return self
261
+ for key, value in other.values.items():
262
+ type = distutils_settings[key]
263
+ if type is transitive_str and key not in self.values:
264
+ self.values[key] = value
265
+ elif type is transitive_list:
266
+ if key in self.values:
267
+ # Change a *copy* of the list (Trac #845)
268
+ all = self.values[key][:]
269
+ for v in value:
270
+ if v not in all:
271
+ all.append(v)
272
+ value = all
273
+ self.values[key] = value
274
+ elif type is bool_or:
275
+ self.values[key] = self.values.get(key, False) | value
276
+ return self
277
+
278
+ def subs(self, aliases):
279
+ if aliases is None:
280
+ return self
281
+ resolved = DistutilsInfo()
282
+ for key, value in self.values.items():
283
+ type = distutils_settings[key]
284
+ if type in [list, transitive_list]:
285
+ new_value_list = []
286
+ for v in value:
287
+ if v in aliases:
288
+ v = aliases[v]
289
+ if isinstance(v, list):
290
+ new_value_list += v
291
+ else:
292
+ new_value_list.append(v)
293
+ value = new_value_list
294
+ else:
295
+ if value in aliases:
296
+ value = aliases[value]
297
+ resolved.values[key] = value
298
+ return resolved
299
+
300
+ def apply(self, extension):
301
+ for key, value in self.values.items():
302
+ type = distutils_settings[key]
303
+ if type in [list, transitive_list]:
304
+ value = getattr(extension, key) + list(value)
305
+ setattr(extension, key, value)
306
+
307
+
308
+ @cython.locals(start=cython.Py_ssize_t, q=cython.Py_ssize_t,
309
+ single_q=cython.Py_ssize_t, double_q=cython.Py_ssize_t,
310
+ hash_mark=cython.Py_ssize_t, end=cython.Py_ssize_t,
311
+ k=cython.Py_ssize_t, counter=cython.Py_ssize_t, quote_len=cython.Py_ssize_t)
312
+ def strip_string_literals(code, prefix='__Pyx_L'):
313
+ """
314
+ Normalizes every string literal to be of the form '__Pyx_Lxxx',
315
+ returning the normalized code and a mapping of labels to
316
+ string literals.
317
+ """
318
+ new_code = []
319
+ literals = {}
320
+ counter = 0
321
+ start = q = 0
322
+ in_quote = False
323
+ hash_mark = single_q = double_q = -1
324
+ code_len = len(code)
325
+ quote_type = None
326
+ quote_len = -1
327
+
328
+ while True:
329
+ if hash_mark < q:
330
+ hash_mark = code.find('#', q)
331
+ if single_q < q:
332
+ single_q = code.find("'", q)
333
+ if double_q < q:
334
+ double_q = code.find('"', q)
335
+ q = min(single_q, double_q)
336
+ if q == -1:
337
+ q = max(single_q, double_q)
338
+
339
+ # We're done.
340
+ if q == -1 and hash_mark == -1:
341
+ new_code.append(code[start:])
342
+ break
343
+
344
+ # Try to close the quote.
345
+ elif in_quote:
346
+ if code[q-1] == u'\\':
347
+ k = 2
348
+ while q >= k and code[q-k] == u'\\':
349
+ k += 1
350
+ if k % 2 == 0:
351
+ q += 1
352
+ continue
353
+ if code[q] == quote_type and (
354
+ quote_len == 1 or (code_len > q + 2 and quote_type == code[q+1] == code[q+2])):
355
+ counter += 1
356
+ label = "%s%s_" % (prefix, counter)
357
+ literals[label] = code[start+quote_len:q]
358
+ full_quote = code[q:q+quote_len]
359
+ new_code.append(full_quote)
360
+ new_code.append(label)
361
+ new_code.append(full_quote)
362
+ q += quote_len
363
+ in_quote = False
364
+ start = q
365
+ else:
366
+ q += 1
367
+
368
+ # Process comment.
369
+ elif -1 != hash_mark and (hash_mark < q or q == -1):
370
+ new_code.append(code[start:hash_mark+1])
371
+ end = code.find('\n', hash_mark)
372
+ counter += 1
373
+ label = "%s%s_" % (prefix, counter)
374
+ if end == -1:
375
+ end_or_none = None
376
+ else:
377
+ end_or_none = end
378
+ literals[label] = code[hash_mark+1:end_or_none]
379
+ new_code.append(label)
380
+ if end == -1:
381
+ break
382
+ start = q = end
383
+
384
+ # Open the quote.
385
+ else:
386
+ if code_len >= q+3 and (code[q] == code[q+1] == code[q+2]):
387
+ quote_len = 3
388
+ else:
389
+ quote_len = 1
390
+ in_quote = True
391
+ quote_type = code[q]
392
+ new_code.append(code[start:q])
393
+ start = q
394
+ q += quote_len
395
+
396
+ return "".join(new_code), literals
397
+
398
+
399
+ # We need to allow spaces to allow for conditional compilation like
400
+ # IF ...:
401
+ # cimport ...
402
+ dependency_regex = re.compile(r"(?:^\s*from +([0-9a-zA-Z_.]+) +cimport)|"
403
+ r"(?:^\s*cimport +([0-9a-zA-Z_.]+(?: *, *[0-9a-zA-Z_.]+)*))|"
404
+ r"(?:^\s*cdef +extern +from +['\"]([^'\"]+)['\"])|"
405
+ r"(?:^\s*include +['\"]([^'\"]+)['\"])", re.M)
406
+ dependency_after_from_regex = re.compile(
407
+ r"(?:^\s+\(([0-9a-zA-Z_., ]*)\)[#\n])|"
408
+ r"(?:^\s+([0-9a-zA-Z_., ]*)[#\n])",
409
+ re.M)
410
+
411
+
412
+ def normalize_existing(base_path, rel_paths):
413
+ return normalize_existing0(os.path.dirname(base_path), tuple(set(rel_paths)))
414
+
415
+
416
+ @cached_function
417
+ def normalize_existing0(base_dir, rel_paths):
418
+ """
419
+ Given some base directory ``base_dir`` and a list of path names
420
+ ``rel_paths``, normalize each relative path name ``rel`` by
421
+ replacing it by ``os.path.join(base, rel)`` if that file exists.
422
+
423
+ Return a couple ``(normalized, needed_base)`` where ``normalized``
424
+ if the list of normalized file names and ``needed_base`` is
425
+ ``base_dir`` if we actually needed ``base_dir``. If no paths were
426
+ changed (for example, if all paths were already absolute), then
427
+ ``needed_base`` is ``None``.
428
+ """
429
+ normalized = []
430
+ needed_base = None
431
+ for rel in rel_paths:
432
+ if os.path.isabs(rel):
433
+ normalized.append(rel)
434
+ continue
435
+ path = join_path(base_dir, rel)
436
+ if path_exists(path):
437
+ normalized.append(os.path.normpath(path))
438
+ needed_base = base_dir
439
+ else:
440
+ normalized.append(rel)
441
+ return (normalized, needed_base)
442
+
443
+
444
+ def resolve_depends(depends, include_dirs):
445
+ include_dirs = tuple(include_dirs)
446
+ resolved = []
447
+ for depend in depends:
448
+ path = resolve_depend(depend, include_dirs)
449
+ if path is not None:
450
+ resolved.append(path)
451
+ return resolved
452
+
453
+
454
+ @cached_function
455
+ def resolve_depend(depend, include_dirs):
456
+ if depend[0] == '<' and depend[-1] == '>':
457
+ return None
458
+ for dir in include_dirs:
459
+ path = join_path(dir, depend)
460
+ if path_exists(path):
461
+ return os.path.normpath(path)
462
+ return None
463
+
464
+
465
+ @cached_function
466
+ def package(filename):
467
+ dir = os.path.dirname(os.path.abspath(str(filename)))
468
+ if dir != filename and is_package_dir(dir):
469
+ return package(dir) + (os.path.basename(dir),)
470
+ else:
471
+ return ()
472
+
473
+
474
+ @cached_function
475
+ def fully_qualified_name(filename):
476
+ module = os.path.splitext(os.path.basename(filename))[0]
477
+ return '.'.join(package(filename) + (module,))
478
+
479
+
480
+ @cached_function
481
+ def parse_dependencies(source_filename):
482
+ # Actual parsing is way too slow, so we use regular expressions.
483
+ # The only catch is that we must strip comments and string
484
+ # literals ahead of time.
485
+ with Utils.open_source_file(source_filename, error_handling='ignore') as fh:
486
+ source = fh.read()
487
+ distutils_info = DistutilsInfo(source)
488
+ source, literals = strip_string_literals(source)
489
+ source = source.replace('\\\n', ' ').replace('\t', ' ')
490
+
491
+ # TODO: pure mode
492
+ cimports = []
493
+ includes = []
494
+ externs = []
495
+ for m in dependency_regex.finditer(source):
496
+ cimport_from, cimport_list, extern, include = m.groups()
497
+ if cimport_from:
498
+ cimports.append(cimport_from)
499
+ m_after_from = dependency_after_from_regex.search(source, pos=m.end())
500
+ if m_after_from:
501
+ multiline, one_line = m_after_from.groups()
502
+ subimports = multiline or one_line
503
+ cimports.extend("{0}.{1}".format(cimport_from, s.strip())
504
+ for s in subimports.split(','))
505
+
506
+ elif cimport_list:
507
+ cimports.extend(x.strip() for x in cimport_list.split(","))
508
+ elif extern:
509
+ externs.append(literals[extern])
510
+ else:
511
+ includes.append(literals[include])
512
+ return cimports, includes, externs, distutils_info
513
+
514
+
515
+ class DependencyTree(object):
516
+
517
+ def __init__(self, context, quiet=False):
518
+ self.context = context
519
+ self.quiet = quiet
520
+ self._transitive_cache = {}
521
+
522
+ def parse_dependencies(self, source_filename):
523
+ if path_exists(source_filename):
524
+ source_filename = os.path.normpath(source_filename)
525
+ return parse_dependencies(source_filename)
526
+
527
+ @cached_method
528
+ def included_files(self, filename):
529
+ # This is messy because included files are textually included, resolving
530
+ # cimports (but not includes) relative to the including file.
531
+ all = set()
532
+ for include in self.parse_dependencies(filename)[1]:
533
+ include_path = join_path(os.path.dirname(filename), include)
534
+ if not path_exists(include_path):
535
+ include_path = self.context.find_include_file(include, None)
536
+ if include_path:
537
+ if '.' + os.path.sep in include_path:
538
+ include_path = os.path.normpath(include_path)
539
+ all.add(include_path)
540
+ all.update(self.included_files(include_path))
541
+ elif not self.quiet:
542
+ print("Unable to locate '%s' referenced from '%s'" % (filename, include))
543
+ return all
544
+
545
+ @cached_method
546
+ def cimports_externs_incdirs(self, filename):
547
+ # This is really ugly. Nested cimports are resolved with respect to the
548
+ # includer, but includes are resolved with respect to the includee.
549
+ cimports, includes, externs = self.parse_dependencies(filename)[:3]
550
+ cimports = set(cimports)
551
+ externs = set(externs)
552
+ incdirs = set()
553
+ for include in self.included_files(filename):
554
+ included_cimports, included_externs, included_incdirs = self.cimports_externs_incdirs(include)
555
+ cimports.update(included_cimports)
556
+ externs.update(included_externs)
557
+ incdirs.update(included_incdirs)
558
+ externs, incdir = normalize_existing(filename, externs)
559
+ if incdir:
560
+ incdirs.add(incdir)
561
+ return tuple(cimports), externs, incdirs
562
+
563
+ def cimports(self, filename):
564
+ return self.cimports_externs_incdirs(filename)[0]
565
+
566
+ def package(self, filename):
567
+ return package(filename)
568
+
569
+ def fully_qualified_name(self, filename):
570
+ return fully_qualified_name(filename)
571
+
572
+ @cached_method
573
+ def find_pxd(self, module, filename=None):
574
+ is_relative = module[0] == '.'
575
+ if is_relative and not filename:
576
+ raise NotImplementedError("New relative imports.")
577
+ if filename is not None:
578
+ module_path = module.split('.')
579
+ if is_relative:
580
+ module_path.pop(0) # just explicitly relative
581
+ package_path = list(self.package(filename))
582
+ while module_path and not module_path[0]:
583
+ try:
584
+ package_path.pop()
585
+ except IndexError:
586
+ return None # FIXME: error?
587
+ module_path.pop(0)
588
+ relative = '.'.join(package_path + module_path)
589
+ pxd = self.context.find_pxd_file(relative, None)
590
+ if pxd:
591
+ return pxd
592
+ if is_relative:
593
+ return None # FIXME: error?
594
+ return self.context.find_pxd_file(module, None)
595
+
596
+ @cached_method
597
+ def cimported_files(self, filename):
598
+ if filename[-4:] == '.pyx' and path_exists(filename[:-4] + '.pxd'):
599
+ pxd_list = [filename[:-4] + '.pxd']
600
+ else:
601
+ pxd_list = []
602
+ # Cimports generates all possible combinations package.module
603
+ # when imported as from package cimport module.
604
+ for module in self.cimports(filename):
605
+ if module[:7] == 'cython.' or module == 'cython':
606
+ continue
607
+ pxd_file = self.find_pxd(module, filename)
608
+ if pxd_file is not None:
609
+ pxd_list.append(pxd_file)
610
+ return tuple(pxd_list)
611
+
612
+ @cached_method
613
+ def immediate_dependencies(self, filename):
614
+ all = set([filename])
615
+ all.update(self.cimported_files(filename))
616
+ all.update(self.included_files(filename))
617
+ return all
618
+
619
+ def all_dependencies(self, filename):
620
+ return self.transitive_merge(filename, self.immediate_dependencies, set.union)
621
+
622
+ @cached_method
623
+ def timestamp(self, filename):
624
+ return os.path.getmtime(filename)
625
+
626
+ def extract_timestamp(self, filename):
627
+ return self.timestamp(filename), filename
628
+
629
+ def newest_dependency(self, filename):
630
+ return max([self.extract_timestamp(f) for f in self.all_dependencies(filename)])
631
+
632
+ def transitive_fingerprint(self, filename, module, compilation_options):
633
+ r"""
634
+ Return a fingerprint of a cython file that is about to be cythonized.
635
+
636
+ Fingerprints are looked up in future compilations. If the fingerprint
637
+ is found, the cythonization can be skipped. The fingerprint must
638
+ incorporate everything that has an influence on the generated code.
639
+ """
640
+ try:
641
+ m = hashlib.md5(__version__.encode('UTF-8'))
642
+ m.update(file_hash(filename).encode('UTF-8'))
643
+ for x in sorted(self.all_dependencies(filename)):
644
+ if os.path.splitext(x)[1] not in ('.c', '.cpp', '.h'):
645
+ m.update(file_hash(x).encode('UTF-8'))
646
+ # Include the module attributes that change the compilation result
647
+ # in the fingerprint. We do not iterate over module.__dict__ and
648
+ # include almost everything here as users might extend Extension
649
+ # with arbitrary (random) attributes that would lead to cache
650
+ # misses.
651
+ m.update(str((
652
+ module.language,
653
+ getattr(module, 'py_limited_api', False),
654
+ getattr(module, 'np_pythran', False)
655
+ )).encode('UTF-8'))
656
+
657
+ m.update(compilation_options.get_fingerprint().encode('UTF-8'))
658
+ return m.hexdigest()
659
+ except IOError:
660
+ return None
661
+
662
+ def distutils_info0(self, filename):
663
+ info = self.parse_dependencies(filename)[3]
664
+ kwds = info.values
665
+ cimports, externs, incdirs = self.cimports_externs_incdirs(filename)
666
+ basedir = os.getcwd()
667
+ # Add dependencies on "cdef extern from ..." files
668
+ if externs:
669
+ externs = _make_relative(externs, basedir)
670
+ if 'depends' in kwds:
671
+ kwds['depends'] = list(set(kwds['depends']).union(externs))
672
+ else:
673
+ kwds['depends'] = list(externs)
674
+ # Add include_dirs to ensure that the C compiler will find the
675
+ # "cdef extern from ..." files
676
+ if incdirs:
677
+ include_dirs = list(kwds.get('include_dirs', []))
678
+ for inc in _make_relative(incdirs, basedir):
679
+ if inc not in include_dirs:
680
+ include_dirs.append(inc)
681
+ kwds['include_dirs'] = include_dirs
682
+ return info
683
+
684
+ def distutils_info(self, filename, aliases=None, base=None):
685
+ return (self.transitive_merge(filename, self.distutils_info0, DistutilsInfo.merge)
686
+ .subs(aliases)
687
+ .merge(base))
688
+
689
+ def transitive_merge(self, node, extract, merge):
690
+ try:
691
+ seen = self._transitive_cache[extract, merge]
692
+ except KeyError:
693
+ seen = self._transitive_cache[extract, merge] = {}
694
+ return self.transitive_merge_helper(
695
+ node, extract, merge, seen, {}, self.cimported_files)[0]
696
+
697
+ def transitive_merge_helper(self, node, extract, merge, seen, stack, outgoing):
698
+ if node in seen:
699
+ return seen[node], None
700
+ deps = extract(node)
701
+ if node in stack:
702
+ return deps, node
703
+ try:
704
+ stack[node] = len(stack)
705
+ loop = None
706
+ for next in outgoing(node):
707
+ sub_deps, sub_loop = self.transitive_merge_helper(next, extract, merge, seen, stack, outgoing)
708
+ if sub_loop is not None:
709
+ if loop is not None and stack[loop] < stack[sub_loop]:
710
+ pass
711
+ else:
712
+ loop = sub_loop
713
+ deps = merge(deps, sub_deps)
714
+ if loop == node:
715
+ loop = None
716
+ if loop is None:
717
+ seen[node] = deps
718
+ return deps, loop
719
+ finally:
720
+ del stack[node]
721
+
722
+
723
+ _dep_tree = None
724
+
725
+ def create_dependency_tree(ctx=None, quiet=False):
726
+ global _dep_tree
727
+ if _dep_tree is None:
728
+ if ctx is None:
729
+ ctx = Context(["."], CompilationOptions(default_options))
730
+ _dep_tree = DependencyTree(ctx, quiet=quiet)
731
+ return _dep_tree
732
+
733
+
734
+ # If this changes, change also docs/src/reference/compilation.rst
735
+ # which mentions this function
736
+ def default_create_extension(template, kwds):
737
+ if 'depends' in kwds:
738
+ include_dirs = kwds.get('include_dirs', []) + ["."]
739
+ depends = resolve_depends(kwds['depends'], include_dirs)
740
+ kwds['depends'] = sorted(set(depends + template.depends))
741
+
742
+ t = template.__class__
743
+ ext = t(**kwds)
744
+ metadata = dict(distutils=kwds, module_name=kwds['name'])
745
+ return (ext, metadata)
746
+
747
+
748
+ # This may be useful for advanced users?
749
+ def create_extension_list(patterns, exclude=None, ctx=None, aliases=None, quiet=False, language=None,
750
+ exclude_failures=False):
751
+ if language is not None:
752
+ print('Warning: passing language={0!r} to cythonize() is deprecated. '
753
+ 'Instead, put "# distutils: language={0}" in your .pyx or .pxd file(s)'.format(language))
754
+ if exclude is None:
755
+ exclude = []
756
+ if patterns is None:
757
+ return [], {}
758
+ elif isinstance(patterns, basestring) or not isinstance(patterns, Iterable):
759
+ patterns = [patterns]
760
+ explicit_modules = set([m.name for m in patterns if isinstance(m, Extension)])
761
+ seen = set()
762
+ deps = create_dependency_tree(ctx, quiet=quiet)
763
+ to_exclude = set()
764
+ if not isinstance(exclude, list):
765
+ exclude = [exclude]
766
+ for pattern in exclude:
767
+ to_exclude.update(map(os.path.abspath, extended_iglob(pattern)))
768
+
769
+ module_list = []
770
+ module_metadata = {}
771
+
772
+ # workaround for setuptools
773
+ if 'setuptools' in sys.modules:
774
+ Extension_distutils = sys.modules['setuptools.extension']._Extension
775
+ Extension_setuptools = sys.modules['setuptools'].Extension
776
+ else:
777
+ # dummy class, in case we do not have setuptools
778
+ Extension_distutils = Extension
779
+ class Extension_setuptools(Extension): pass
780
+
781
+ # if no create_extension() function is defined, use a simple
782
+ # default function.
783
+ create_extension = ctx.options.create_extension or default_create_extension
784
+
785
+ for pattern in patterns:
786
+ if isinstance(pattern, str):
787
+ filepattern = pattern
788
+ template = Extension(pattern, []) # Fake Extension without sources
789
+ name = '*'
790
+ base = None
791
+ ext_language = language
792
+ elif isinstance(pattern, (Extension_distutils, Extension_setuptools)):
793
+ cython_sources = [s for s in pattern.sources
794
+ if os.path.splitext(s)[1] in ('.py', '.pyx')]
795
+ if cython_sources:
796
+ filepattern = cython_sources[0]
797
+ if len(cython_sources) > 1:
798
+ print("Warning: Multiple cython sources found for extension '%s': %s\n"
799
+ "See http://cython.readthedocs.io/en/latest/src/userguide/sharing_declarations.html "
800
+ "for sharing declarations among Cython files." % (pattern.name, cython_sources))
801
+ else:
802
+ # ignore non-cython modules
803
+ module_list.append(pattern)
804
+ continue
805
+ template = pattern
806
+ name = template.name
807
+ base = DistutilsInfo(exn=template)
808
+ ext_language = None # do not override whatever the Extension says
809
+ else:
810
+ msg = str("pattern is not of type str nor subclass of Extension (%s)"
811
+ " but of type %s and class %s" % (repr(Extension),
812
+ type(pattern),
813
+ pattern.__class__))
814
+ raise TypeError(msg)
815
+
816
+ for file in nonempty(sorted(extended_iglob(filepattern)), "'%s' doesn't match any files" % filepattern):
817
+ if os.path.abspath(file) in to_exclude:
818
+ continue
819
+ module_name = deps.fully_qualified_name(file)
820
+ if '*' in name:
821
+ if module_name in explicit_modules:
822
+ continue
823
+ elif name:
824
+ module_name = name
825
+
826
+ Utils.raise_error_if_module_name_forbidden(module_name)
827
+
828
+ if module_name not in seen:
829
+ try:
830
+ kwds = deps.distutils_info(file, aliases, base).values
831
+ except Exception:
832
+ if exclude_failures:
833
+ continue
834
+ raise
835
+ if base is not None:
836
+ for key, value in base.values.items():
837
+ if key not in kwds:
838
+ kwds[key] = value
839
+
840
+ kwds['name'] = module_name
841
+
842
+ sources = [file] + [m for m in template.sources if m != filepattern]
843
+ if 'sources' in kwds:
844
+ # allow users to add .c files etc.
845
+ for source in kwds['sources']:
846
+ source = encode_filename_in_py2(source)
847
+ if source not in sources:
848
+ sources.append(source)
849
+ kwds['sources'] = sources
850
+
851
+ if ext_language and 'language' not in kwds:
852
+ kwds['language'] = ext_language
853
+
854
+ np_pythran = kwds.pop('np_pythran', False)
855
+
856
+ # Create the new extension
857
+ m, metadata = create_extension(template, kwds)
858
+ m.np_pythran = np_pythran or getattr(m, 'np_pythran', False)
859
+ if m.np_pythran:
860
+ update_pythran_extension(m)
861
+ module_list.append(m)
862
+
863
+ # Store metadata (this will be written as JSON in the
864
+ # generated C file but otherwise has no purpose)
865
+ module_metadata[module_name] = metadata
866
+
867
+ if file not in m.sources:
868
+ # Old setuptools unconditionally replaces .pyx with .c/.cpp
869
+ target_file = os.path.splitext(file)[0] + ('.cpp' if m.language == 'c++' else '.c')
870
+ try:
871
+ m.sources.remove(target_file)
872
+ except ValueError:
873
+ # never seen this in the wild, but probably better to warn about this unexpected case
874
+ print("Warning: Cython source file not found in sources list, adding %s" % file)
875
+ m.sources.insert(0, file)
876
+ seen.add(name)
877
+ return module_list, module_metadata
878
+
879
+
880
+ # This is the user-exposed entry point.
881
+ def cythonize(module_list, exclude=None, nthreads=0, aliases=None, quiet=False, force=False, language=None,
882
+ exclude_failures=False, **options):
883
+ """
884
+ Compile a set of source modules into C/C++ files and return a list of distutils
885
+ Extension objects for them.
886
+
887
+ :param module_list: As module list, pass either a glob pattern, a list of glob
888
+ patterns or a list of Extension objects. The latter
889
+ allows you to configure the extensions separately
890
+ through the normal distutils options.
891
+ You can also pass Extension objects that have
892
+ glob patterns as their sources. Then, cythonize
893
+ will resolve the pattern and create a
894
+ copy of the Extension for every matching file.
895
+
896
+ :param exclude: When passing glob patterns as ``module_list``, you can exclude certain
897
+ module names explicitly by passing them into the ``exclude`` option.
898
+
899
+ :param nthreads: The number of concurrent builds for parallel compilation
900
+ (requires the ``multiprocessing`` module).
901
+
902
+ :param aliases: If you want to use compiler directives like ``# distutils: ...`` but
903
+ can only know at compile time (when running the ``setup.py``) which values
904
+ to use, you can use aliases and pass a dictionary mapping those aliases
905
+ to Python strings when calling :func:`cythonize`. As an example, say you
906
+ want to use the compiler
907
+ directive ``# distutils: include_dirs = ../static_libs/include/``
908
+ but this path isn't always fixed and you want to find it when running
909
+ the ``setup.py``. You can then do ``# distutils: include_dirs = MY_HEADERS``,
910
+ find the value of ``MY_HEADERS`` in the ``setup.py``, put it in a python
911
+ variable called ``foo`` as a string, and then call
912
+ ``cythonize(..., aliases={'MY_HEADERS': foo})``.
913
+
914
+ :param quiet: If True, Cython won't print error, warning, or status messages during the
915
+ compilation.
916
+
917
+ :param force: Forces the recompilation of the Cython modules, even if the timestamps
918
+ don't indicate that a recompilation is necessary.
919
+
920
+ :param language: To globally enable C++ mode, you can pass ``language='c++'``. Otherwise, this
921
+ will be determined at a per-file level based on compiler directives. This
922
+ affects only modules found based on file names. Extension instances passed
923
+ into :func:`cythonize` will not be changed. It is recommended to rather
924
+ use the compiler directive ``# distutils: language = c++`` than this option.
925
+
926
+ :param exclude_failures: For a broad 'try to compile' mode that ignores compilation
927
+ failures and simply excludes the failed extensions,
928
+ pass ``exclude_failures=True``. Note that this only
929
+ really makes sense for compiling ``.py`` files which can also
930
+ be used without compilation.
931
+
932
+ :param annotate: If ``True``, will produce a HTML file for each of the ``.pyx`` or ``.py``
933
+ files compiled. The HTML file gives an indication
934
+ of how much Python interaction there is in
935
+ each of the source code lines, compared to plain C code.
936
+ It also allows you to see the C/C++ code
937
+ generated for each line of Cython code. This report is invaluable when
938
+ optimizing a function for speed,
939
+ and for determining when to :ref:`release the GIL <nogil>`:
940
+ in general, a ``nogil`` block may contain only "white" code.
941
+ See examples in :ref:`determining_where_to_add_types` or
942
+ :ref:`primes`.
943
+
944
+ :param compiler_directives: Allow to set compiler directives in the ``setup.py`` like this:
945
+ ``compiler_directives={'embedsignature': True}``.
946
+ See :ref:`compiler-directives`.
947
+
948
+ :param depfile: produce depfiles for the sources if True.
949
+ """
950
+ if exclude is None:
951
+ exclude = []
952
+ if 'include_path' not in options:
953
+ options['include_path'] = ['.']
954
+ if 'common_utility_include_dir' in options:
955
+ safe_makedirs(options['common_utility_include_dir'])
956
+
957
+ depfile = options.pop('depfile', None)
958
+
959
+ if pythran is None:
960
+ pythran_options = None
961
+ else:
962
+ pythran_options = CompilationOptions(**options)
963
+ pythran_options.cplus = True
964
+ pythran_options.np_pythran = True
965
+
966
+ c_options = CompilationOptions(**options)
967
+ cpp_options = CompilationOptions(**options); cpp_options.cplus = True
968
+ ctx = c_options.create_context()
969
+ options = c_options
970
+ module_list, module_metadata = create_extension_list(
971
+ module_list,
972
+ exclude=exclude,
973
+ ctx=ctx,
974
+ quiet=quiet,
975
+ exclude_failures=exclude_failures,
976
+ language=language,
977
+ aliases=aliases)
978
+ deps = create_dependency_tree(ctx, quiet=quiet)
979
+ build_dir = getattr(options, 'build_dir', None)
980
+
981
+ def copy_to_build_dir(filepath, root=os.getcwd()):
982
+ filepath_abs = os.path.abspath(filepath)
983
+ if os.path.isabs(filepath):
984
+ filepath = filepath_abs
985
+ if filepath_abs.startswith(root):
986
+ # distutil extension depends are relative to cwd
987
+ mod_dir = join_path(build_dir,
988
+ os.path.dirname(_relpath(filepath, root)))
989
+ copy_once_if_newer(filepath_abs, mod_dir)
990
+
991
+ modules_by_cfile = collections.defaultdict(list)
992
+ to_compile = []
993
+ for m in module_list:
994
+ if build_dir:
995
+ for dep in m.depends:
996
+ copy_to_build_dir(dep)
997
+
998
+ cy_sources = [
999
+ source for source in m.sources
1000
+ if os.path.splitext(source)[1] in ('.pyx', '.py')]
1001
+ if len(cy_sources) == 1:
1002
+ # normal "special" case: believe the Extension module name to allow user overrides
1003
+ full_module_name = m.name
1004
+ else:
1005
+ # infer FQMN from source files
1006
+ full_module_name = None
1007
+
1008
+ new_sources = []
1009
+ for source in m.sources:
1010
+ base, ext = os.path.splitext(source)
1011
+ if ext in ('.pyx', '.py'):
1012
+ if m.np_pythran:
1013
+ c_file = base + '.cpp'
1014
+ options = pythran_options
1015
+ elif m.language == 'c++':
1016
+ c_file = base + '.cpp'
1017
+ options = cpp_options
1018
+ else:
1019
+ c_file = base + '.c'
1020
+ options = c_options
1021
+
1022
+ # setup for out of place build directory if enabled
1023
+ if build_dir:
1024
+ if os.path.isabs(c_file):
1025
+ warnings.warn("build_dir has no effect for absolute source paths")
1026
+ c_file = os.path.join(build_dir, c_file)
1027
+ dir = os.path.dirname(c_file)
1028
+ safe_makedirs_once(dir)
1029
+
1030
+ # write out the depfile, if requested
1031
+ if depfile:
1032
+ dependencies = deps.all_dependencies(source)
1033
+ src_base_dir, _ = os.path.split(source)
1034
+ if not src_base_dir.endswith(os.sep):
1035
+ src_base_dir += os.sep
1036
+ # paths below the base_dir are relative, otherwise absolute
1037
+ paths = []
1038
+ for fname in dependencies:
1039
+ if (fname.startswith(src_base_dir) or
1040
+ fname.startswith('.' + os.path.sep)):
1041
+ paths.append(os.path.relpath(fname, src_base_dir))
1042
+ else:
1043
+ paths.append(os.path.abspath(fname))
1044
+
1045
+ depline = os.path.split(c_file)[1] + ": \\\n "
1046
+ depline += " \\\n ".join(paths) + "\n"
1047
+ with open(c_file+'.dep', 'w') as outfile:
1048
+ outfile.write(depline)
1049
+
1050
+ if os.path.exists(c_file):
1051
+ c_timestamp = os.path.getmtime(c_file)
1052
+ else:
1053
+ c_timestamp = -1
1054
+
1055
+ # Priority goes first to modified files, second to direct
1056
+ # dependents, and finally to indirect dependents.
1057
+ if c_timestamp < deps.timestamp(source):
1058
+ dep_timestamp, dep = deps.timestamp(source), source
1059
+ priority = 0
1060
+ else:
1061
+ dep_timestamp, dep = deps.newest_dependency(source)
1062
+ priority = 2 - (dep in deps.immediate_dependencies(source))
1063
+ if force or c_timestamp < dep_timestamp:
1064
+ if not quiet and not force:
1065
+ if source == dep:
1066
+ print("Compiling %s because it changed." % source)
1067
+ else:
1068
+ print("Compiling %s because it depends on %s." % (source, dep))
1069
+ if not force and options.cache:
1070
+ fingerprint = deps.transitive_fingerprint(source, m, options)
1071
+ else:
1072
+ fingerprint = None
1073
+ to_compile.append((
1074
+ priority, source, c_file, fingerprint, quiet,
1075
+ options, not exclude_failures, module_metadata.get(m.name),
1076
+ full_module_name))
1077
+ new_sources.append(c_file)
1078
+ modules_by_cfile[c_file].append(m)
1079
+ else:
1080
+ new_sources.append(source)
1081
+ if build_dir:
1082
+ copy_to_build_dir(source)
1083
+ m.sources = new_sources
1084
+
1085
+ if options.cache:
1086
+ if not os.path.exists(options.cache):
1087
+ os.makedirs(options.cache)
1088
+ to_compile.sort()
1089
+ # Drop "priority" component of "to_compile" entries and add a
1090
+ # simple progress indicator.
1091
+ N = len(to_compile)
1092
+ progress_fmt = "[{0:%d}/{1}] " % len(str(N))
1093
+ for i in range(N):
1094
+ progress = progress_fmt.format(i+1, N)
1095
+ to_compile[i] = to_compile[i][1:] + (progress,)
1096
+
1097
+ if N <= 1:
1098
+ nthreads = 0
1099
+ if nthreads:
1100
+ # Requires multiprocessing (or Python >= 2.6)
1101
+ try:
1102
+ import multiprocessing
1103
+ pool = multiprocessing.Pool(
1104
+ nthreads, initializer=_init_multiprocessing_helper)
1105
+ except (ImportError, OSError):
1106
+ print("multiprocessing required for parallel cythonization")
1107
+ nthreads = 0
1108
+ else:
1109
+ # This is a bit more involved than it should be, because KeyboardInterrupts
1110
+ # break the multiprocessing workers when using a normal pool.map().
1111
+ # See, for example:
1112
+ # http://noswap.com/blog/python-multiprocessing-keyboardinterrupt
1113
+ try:
1114
+ result = pool.map_async(cythonize_one_helper, to_compile, chunksize=1)
1115
+ pool.close()
1116
+ while not result.ready():
1117
+ try:
1118
+ result.get(99999) # seconds
1119
+ except multiprocessing.TimeoutError:
1120
+ pass
1121
+ except KeyboardInterrupt:
1122
+ pool.terminate()
1123
+ raise
1124
+ pool.join()
1125
+ if not nthreads:
1126
+ for args in to_compile:
1127
+ cythonize_one(*args)
1128
+
1129
+ if exclude_failures:
1130
+ failed_modules = set()
1131
+ for c_file, modules in modules_by_cfile.items():
1132
+ if not os.path.exists(c_file):
1133
+ failed_modules.update(modules)
1134
+ elif os.path.getsize(c_file) < 200:
1135
+ f = io_open(c_file, 'r', encoding='iso8859-1')
1136
+ try:
1137
+ if f.read(len('#error ')) == '#error ':
1138
+ # dead compilation result
1139
+ failed_modules.update(modules)
1140
+ finally:
1141
+ f.close()
1142
+ if failed_modules:
1143
+ for module in failed_modules:
1144
+ module_list.remove(module)
1145
+ print("Failed compilations: %s" % ', '.join(sorted([
1146
+ module.name for module in failed_modules])))
1147
+
1148
+ if options.cache:
1149
+ cleanup_cache(options.cache, getattr(options, 'cache_size', 1024 * 1024 * 100))
1150
+ # cythonize() is often followed by the (non-Python-buffered)
1151
+ # compiler output, flush now to avoid interleaving output.
1152
+ sys.stdout.flush()
1153
+ return module_list
1154
+
1155
+
1156
+ if os.environ.get('XML_RESULTS'):
1157
+ compile_result_dir = os.environ['XML_RESULTS']
1158
+ def record_results(func):
1159
+ def with_record(*args):
1160
+ t = time.time()
1161
+ success = True
1162
+ try:
1163
+ try:
1164
+ func(*args)
1165
+ except:
1166
+ success = False
1167
+ finally:
1168
+ t = time.time() - t
1169
+ module = fully_qualified_name(args[0])
1170
+ name = "cythonize." + module
1171
+ failures = 1 - success
1172
+ if success:
1173
+ failure_item = ""
1174
+ else:
1175
+ failure_item = "failure"
1176
+ output = open(os.path.join(compile_result_dir, name + ".xml"), "w")
1177
+ output.write("""
1178
+ <?xml version="1.0" ?>
1179
+ <testsuite name="%(name)s" errors="0" failures="%(failures)s" tests="1" time="%(t)s">
1180
+ <testcase classname="%(name)s" name="cythonize">
1181
+ %(failure_item)s
1182
+ </testcase>
1183
+ </testsuite>
1184
+ """.strip() % locals())
1185
+ output.close()
1186
+ return with_record
1187
+ else:
1188
+ def record_results(func):
1189
+ return func
1190
+
1191
+
1192
+ # TODO: Share context? Issue: pyx processing leaks into pxd module
1193
+ @record_results
1194
+ def cythonize_one(pyx_file, c_file, fingerprint, quiet, options=None,
1195
+ raise_on_failure=True, embedded_metadata=None, full_module_name=None,
1196
+ progress=""):
1197
+ from ..Compiler.Main import compile_single, default_options
1198
+ from ..Compiler.Errors import CompileError, PyrexError
1199
+
1200
+ if fingerprint:
1201
+ if not os.path.exists(options.cache):
1202
+ safe_makedirs(options.cache)
1203
+ # Cython-generated c files are highly compressible.
1204
+ # (E.g. a compression ratio of about 10 for Sage).
1205
+ fingerprint_file_base = join_path(
1206
+ options.cache, "%s-%s" % (os.path.basename(c_file), fingerprint))
1207
+ gz_fingerprint_file = fingerprint_file_base + gzip_ext
1208
+ zip_fingerprint_file = fingerprint_file_base + '.zip'
1209
+ if os.path.exists(gz_fingerprint_file) or os.path.exists(zip_fingerprint_file):
1210
+ if not quiet:
1211
+ print("%sFound compiled %s in cache" % (progress, pyx_file))
1212
+ if os.path.exists(gz_fingerprint_file):
1213
+ os.utime(gz_fingerprint_file, None)
1214
+ with contextlib.closing(gzip_open(gz_fingerprint_file, 'rb')) as g:
1215
+ with contextlib.closing(open(c_file, 'wb')) as f:
1216
+ shutil.copyfileobj(g, f)
1217
+ else:
1218
+ os.utime(zip_fingerprint_file, None)
1219
+ dirname = os.path.dirname(c_file)
1220
+ with contextlib.closing(zipfile.ZipFile(zip_fingerprint_file)) as z:
1221
+ for artifact in z.namelist():
1222
+ z.extract(artifact, os.path.join(dirname, artifact))
1223
+ return
1224
+ if not quiet:
1225
+ print("%sCythonizing %s" % (progress, pyx_file))
1226
+ if options is None:
1227
+ options = CompilationOptions(default_options)
1228
+ options.output_file = c_file
1229
+ options.embedded_metadata = embedded_metadata
1230
+
1231
+ any_failures = 0
1232
+ try:
1233
+ result = compile_single(pyx_file, options, full_module_name=full_module_name)
1234
+ if result.num_errors > 0:
1235
+ any_failures = 1
1236
+ except (EnvironmentError, PyrexError) as e:
1237
+ sys.stderr.write('%s\n' % e)
1238
+ any_failures = 1
1239
+ # XXX
1240
+ import traceback
1241
+ traceback.print_exc()
1242
+ except Exception:
1243
+ if raise_on_failure:
1244
+ raise
1245
+ import traceback
1246
+ traceback.print_exc()
1247
+ any_failures = 1
1248
+ if any_failures:
1249
+ if raise_on_failure:
1250
+ raise CompileError(None, pyx_file)
1251
+ elif os.path.exists(c_file):
1252
+ os.remove(c_file)
1253
+ elif fingerprint:
1254
+ artifacts = list(filter(None, [
1255
+ getattr(result, attr, None)
1256
+ for attr in ('c_file', 'h_file', 'api_file', 'i_file')]))
1257
+ if len(artifacts) == 1:
1258
+ fingerprint_file = gz_fingerprint_file
1259
+ with contextlib.closing(open(c_file, 'rb')) as f:
1260
+ with contextlib.closing(gzip_open(fingerprint_file + '.tmp', 'wb')) as g:
1261
+ shutil.copyfileobj(f, g)
1262
+ else:
1263
+ fingerprint_file = zip_fingerprint_file
1264
+ with contextlib.closing(zipfile.ZipFile(
1265
+ fingerprint_file + '.tmp', 'w', zipfile_compression_mode)) as zip:
1266
+ for artifact in artifacts:
1267
+ zip.write(artifact, os.path.basename(artifact))
1268
+ os.rename(fingerprint_file + '.tmp', fingerprint_file)
1269
+
1270
+
1271
+ def cythonize_one_helper(m):
1272
+ import traceback
1273
+ try:
1274
+ return cythonize_one(*m)
1275
+ except Exception:
1276
+ traceback.print_exc()
1277
+ raise
1278
+
1279
+
1280
+ def _init_multiprocessing_helper():
1281
+ # KeyboardInterrupt kills workers, so don't let them get it
1282
+ import signal
1283
+ signal.signal(signal.SIGINT, signal.SIG_IGN)
1284
+
1285
+
1286
+ def cleanup_cache(cache, target_size, ratio=.85):
1287
+ try:
1288
+ p = subprocess.Popen(['du', '-s', '-k', os.path.abspath(cache)], stdout=subprocess.PIPE)
1289
+ res = p.wait()
1290
+ if res == 0:
1291
+ total_size = 1024 * int(p.stdout.read().strip().split()[0])
1292
+ if total_size < target_size:
1293
+ return
1294
+ except (OSError, ValueError):
1295
+ pass
1296
+ total_size = 0
1297
+ all = []
1298
+ for file in os.listdir(cache):
1299
+ path = join_path(cache, file)
1300
+ s = os.stat(path)
1301
+ total_size += s.st_size
1302
+ all.append((s.st_atime, s.st_size, path))
1303
+ if total_size > target_size:
1304
+ for time, size, file in reversed(sorted(all)):
1305
+ os.unlink(file)
1306
+ total_size -= size
1307
+ if total_size < target_size * ratio:
1308
+ break
Cython/Build/Distutils.py ADDED
@@ -0,0 +1 @@
 
 
1
+ from Cython.Distutils.build_ext import build_ext
Cython/Build/Inline.py ADDED
@@ -0,0 +1,376 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ import hashlib
4
+ import inspect
5
+ import os
6
+ import re
7
+ import sys
8
+
9
+ from distutils.core import Distribution, Extension
10
+ from distutils.command.build_ext import build_ext
11
+
12
+ import Cython
13
+ from ..Compiler.Main import Context, default_options
14
+
15
+ from ..Compiler.Visitor import CythonTransform, EnvTransform
16
+ from ..Compiler.ParseTreeTransforms import SkipDeclarations
17
+ from ..Compiler.TreeFragment import parse_from_strings
18
+ from ..Compiler.StringEncoding import _unicode
19
+ from .Dependencies import strip_string_literals, cythonize, cached_function
20
+ from ..Compiler import Pipeline
21
+ from ..Utils import get_cython_cache_dir
22
+ import cython as cython_module
23
+
24
+
25
+ IS_PY3 = sys.version_info >= (3,)
26
+
27
+ # A utility function to convert user-supplied ASCII strings to unicode.
28
+ if not IS_PY3:
29
+ def to_unicode(s):
30
+ if isinstance(s, bytes):
31
+ return s.decode('ascii')
32
+ else:
33
+ return s
34
+ else:
35
+ to_unicode = lambda x: x
36
+
37
+ if sys.version_info < (3, 5):
38
+ import imp
39
+ def load_dynamic(name, module_path):
40
+ return imp.load_dynamic(name, module_path)
41
+ else:
42
+ import importlib.util as _importlib_util
43
+ def load_dynamic(name, module_path):
44
+ spec = _importlib_util.spec_from_file_location(name, module_path)
45
+ module = _importlib_util.module_from_spec(spec)
46
+ # sys.modules[name] = module
47
+ spec.loader.exec_module(module)
48
+ return module
49
+
50
+ class UnboundSymbols(EnvTransform, SkipDeclarations):
51
+ def __init__(self):
52
+ CythonTransform.__init__(self, None)
53
+ self.unbound = set()
54
+ def visit_NameNode(self, node):
55
+ if not self.current_env().lookup(node.name):
56
+ self.unbound.add(node.name)
57
+ return node
58
+ def __call__(self, node):
59
+ super(UnboundSymbols, self).__call__(node)
60
+ return self.unbound
61
+
62
+
63
+ @cached_function
64
+ def unbound_symbols(code, context=None):
65
+ code = to_unicode(code)
66
+ if context is None:
67
+ context = Context([], default_options)
68
+ from ..Compiler.ParseTreeTransforms import AnalyseDeclarationsTransform
69
+ tree = parse_from_strings('(tree fragment)', code)
70
+ for phase in Pipeline.create_pipeline(context, 'pyx'):
71
+ if phase is None:
72
+ continue
73
+ tree = phase(tree)
74
+ if isinstance(phase, AnalyseDeclarationsTransform):
75
+ break
76
+ try:
77
+ import builtins
78
+ except ImportError:
79
+ import __builtin__ as builtins
80
+ return tuple(UnboundSymbols()(tree) - set(dir(builtins)))
81
+
82
+
83
+ def unsafe_type(arg, context=None):
84
+ py_type = type(arg)
85
+ if py_type is int:
86
+ return 'long'
87
+ else:
88
+ return safe_type(arg, context)
89
+
90
+
91
+ def safe_type(arg, context=None):
92
+ py_type = type(arg)
93
+ if py_type in (list, tuple, dict, str):
94
+ return py_type.__name__
95
+ elif py_type is complex:
96
+ return 'double complex'
97
+ elif py_type is float:
98
+ return 'double'
99
+ elif py_type is bool:
100
+ return 'bint'
101
+ elif 'numpy' in sys.modules and isinstance(arg, sys.modules['numpy'].ndarray):
102
+ return 'numpy.ndarray[numpy.%s_t, ndim=%s]' % (arg.dtype.name, arg.ndim)
103
+ else:
104
+ for base_type in py_type.__mro__:
105
+ if base_type.__module__ in ('__builtin__', 'builtins'):
106
+ return 'object'
107
+ module = context.find_module(base_type.__module__, need_pxd=False)
108
+ if module:
109
+ entry = module.lookup(base_type.__name__)
110
+ if entry.is_type:
111
+ return '%s.%s' % (base_type.__module__, base_type.__name__)
112
+ return 'object'
113
+
114
+
115
+ def _get_build_extension():
116
+ dist = Distribution()
117
+ # Ensure the build respects distutils configuration by parsing
118
+ # the configuration files
119
+ config_files = dist.find_config_files()
120
+ dist.parse_config_files(config_files)
121
+ build_extension = build_ext(dist)
122
+ build_extension.finalize_options()
123
+ return build_extension
124
+
125
+
126
+ @cached_function
127
+ def _create_context(cython_include_dirs):
128
+ return Context(list(cython_include_dirs), default_options)
129
+
130
+
131
+ _cython_inline_cache = {}
132
+ _cython_inline_default_context = _create_context(('.',))
133
+
134
+
135
+ def _populate_unbound(kwds, unbound_symbols, locals=None, globals=None):
136
+ for symbol in unbound_symbols:
137
+ if symbol not in kwds:
138
+ if locals is None or globals is None:
139
+ calling_frame = inspect.currentframe().f_back.f_back.f_back
140
+ if locals is None:
141
+ locals = calling_frame.f_locals
142
+ if globals is None:
143
+ globals = calling_frame.f_globals
144
+ if symbol in locals:
145
+ kwds[symbol] = locals[symbol]
146
+ elif symbol in globals:
147
+ kwds[symbol] = globals[symbol]
148
+ else:
149
+ print("Couldn't find %r" % symbol)
150
+
151
+
152
+ def _inline_key(orig_code, arg_sigs, language_level):
153
+ key = orig_code, arg_sigs, sys.version_info, sys.executable, language_level, Cython.__version__
154
+ return hashlib.sha1(_unicode(key).encode('utf-8')).hexdigest()
155
+
156
+
157
+ def cython_inline(code, get_type=unsafe_type,
158
+ lib_dir=os.path.join(get_cython_cache_dir(), 'inline'),
159
+ cython_include_dirs=None, cython_compiler_directives=None,
160
+ force=False, quiet=False, locals=None, globals=None, language_level=None, **kwds):
161
+
162
+ if get_type is None:
163
+ get_type = lambda x: 'object'
164
+ ctx = _create_context(tuple(cython_include_dirs)) if cython_include_dirs else _cython_inline_default_context
165
+
166
+ cython_compiler_directives = dict(cython_compiler_directives) if cython_compiler_directives else {}
167
+ if language_level is None and 'language_level' not in cython_compiler_directives:
168
+ language_level = '3str'
169
+ if language_level is not None:
170
+ cython_compiler_directives['language_level'] = language_level
171
+
172
+ # Fast path if this has been called in this session.
173
+ _unbound_symbols = _cython_inline_cache.get(code)
174
+ if _unbound_symbols is not None:
175
+ _populate_unbound(kwds, _unbound_symbols, locals, globals)
176
+ args = sorted(kwds.items())
177
+ arg_sigs = tuple([(get_type(value, ctx), arg) for arg, value in args])
178
+ key_hash = _inline_key(code, arg_sigs, language_level)
179
+ invoke = _cython_inline_cache.get((code, arg_sigs, key_hash))
180
+ if invoke is not None:
181
+ arg_list = [arg[1] for arg in args]
182
+ return invoke(*arg_list)
183
+
184
+ orig_code = code
185
+ code = to_unicode(code)
186
+ code, literals = strip_string_literals(code)
187
+ code = strip_common_indent(code)
188
+ if locals is None:
189
+ locals = inspect.currentframe().f_back.f_back.f_locals
190
+ if globals is None:
191
+ globals = inspect.currentframe().f_back.f_back.f_globals
192
+ try:
193
+ _cython_inline_cache[orig_code] = _unbound_symbols = unbound_symbols(code)
194
+ _populate_unbound(kwds, _unbound_symbols, locals, globals)
195
+ except AssertionError:
196
+ if not quiet:
197
+ # Parsing from strings not fully supported (e.g. cimports).
198
+ print("Could not parse code as a string (to extract unbound symbols).")
199
+
200
+ cimports = []
201
+ for name, arg in list(kwds.items()):
202
+ if arg is cython_module:
203
+ cimports.append('\ncimport cython as %s' % name)
204
+ del kwds[name]
205
+ arg_names = sorted(kwds)
206
+ arg_sigs = tuple([(get_type(kwds[arg], ctx), arg) for arg in arg_names])
207
+ key_hash = _inline_key(orig_code, arg_sigs, language_level)
208
+ module_name = "_cython_inline_" + key_hash
209
+
210
+ if module_name in sys.modules:
211
+ module = sys.modules[module_name]
212
+
213
+ else:
214
+ build_extension = None
215
+ if cython_inline.so_ext is None:
216
+ # Figure out and cache current extension suffix
217
+ build_extension = _get_build_extension()
218
+ cython_inline.so_ext = build_extension.get_ext_filename('')
219
+
220
+ module_path = os.path.join(lib_dir, module_name + cython_inline.so_ext)
221
+
222
+ if not os.path.exists(lib_dir):
223
+ os.makedirs(lib_dir)
224
+ if force or not os.path.isfile(module_path):
225
+ cflags = []
226
+ c_include_dirs = []
227
+ qualified = re.compile(r'([.\w]+)[.]')
228
+ for type, _ in arg_sigs:
229
+ m = qualified.match(type)
230
+ if m:
231
+ cimports.append('\ncimport %s' % m.groups()[0])
232
+ # one special case
233
+ if m.groups()[0] == 'numpy':
234
+ import numpy
235
+ c_include_dirs.append(numpy.get_include())
236
+ # cflags.append('-Wno-unused')
237
+ module_body, func_body = extract_func_code(code)
238
+ params = ', '.join(['%s %s' % a for a in arg_sigs])
239
+ module_code = """
240
+ %(module_body)s
241
+ %(cimports)s
242
+ def __invoke(%(params)s):
243
+ %(func_body)s
244
+ return locals()
245
+ """ % {'cimports': '\n'.join(cimports),
246
+ 'module_body': module_body,
247
+ 'params': params,
248
+ 'func_body': func_body }
249
+ for key, value in literals.items():
250
+ module_code = module_code.replace(key, value)
251
+ pyx_file = os.path.join(lib_dir, module_name + '.pyx')
252
+ fh = open(pyx_file, 'w')
253
+ try:
254
+ fh.write(module_code)
255
+ finally:
256
+ fh.close()
257
+ extension = Extension(
258
+ name = module_name,
259
+ sources = [pyx_file],
260
+ include_dirs = c_include_dirs,
261
+ extra_compile_args = cflags)
262
+ if build_extension is None:
263
+ build_extension = _get_build_extension()
264
+ build_extension.extensions = cythonize(
265
+ [extension],
266
+ include_path=cython_include_dirs or ['.'],
267
+ compiler_directives=cython_compiler_directives,
268
+ quiet=quiet)
269
+ build_extension.build_temp = os.path.dirname(pyx_file)
270
+ build_extension.build_lib = lib_dir
271
+ build_extension.run()
272
+
273
+ module = load_dynamic(module_name, module_path)
274
+
275
+ _cython_inline_cache[orig_code, arg_sigs, key_hash] = module.__invoke
276
+ arg_list = [kwds[arg] for arg in arg_names]
277
+ return module.__invoke(*arg_list)
278
+
279
+
280
+ # Cached suffix used by cython_inline above. None should get
281
+ # overridden with actual value upon the first cython_inline invocation
282
+ cython_inline.so_ext = None
283
+
284
+ _find_non_space = re.compile('[^ ]').search
285
+
286
+
287
+ def strip_common_indent(code):
288
+ min_indent = None
289
+ lines = code.splitlines()
290
+ for line in lines:
291
+ match = _find_non_space(line)
292
+ if not match:
293
+ continue # blank
294
+ indent = match.start()
295
+ if line[indent] == '#':
296
+ continue # comment
297
+ if min_indent is None or min_indent > indent:
298
+ min_indent = indent
299
+ for ix, line in enumerate(lines):
300
+ match = _find_non_space(line)
301
+ if not match or not line or line[indent:indent+1] == '#':
302
+ continue
303
+ lines[ix] = line[min_indent:]
304
+ return '\n'.join(lines)
305
+
306
+
307
+ module_statement = re.compile(r'^((cdef +(extern|class))|cimport|(from .+ cimport)|(from .+ import +[*]))')
308
+ def extract_func_code(code):
309
+ module = []
310
+ function = []
311
+ current = function
312
+ code = code.replace('\t', ' ')
313
+ lines = code.split('\n')
314
+ for line in lines:
315
+ if not line.startswith(' '):
316
+ if module_statement.match(line):
317
+ current = module
318
+ else:
319
+ current = function
320
+ current.append(line)
321
+ return '\n'.join(module), ' ' + '\n '.join(function)
322
+
323
+
324
+ try:
325
+ from inspect import getcallargs
326
+ except ImportError:
327
+ def getcallargs(func, *arg_values, **kwd_values):
328
+ all = {}
329
+ args, varargs, kwds, defaults = inspect.getargspec(func)
330
+ if varargs is not None:
331
+ all[varargs] = arg_values[len(args):]
332
+ for name, value in zip(args, arg_values):
333
+ all[name] = value
334
+ for name, value in list(kwd_values.items()):
335
+ if name in args:
336
+ if name in all:
337
+ raise TypeError("Duplicate argument %s" % name)
338
+ all[name] = kwd_values.pop(name)
339
+ if kwds is not None:
340
+ all[kwds] = kwd_values
341
+ elif kwd_values:
342
+ raise TypeError("Unexpected keyword arguments: %s" % list(kwd_values))
343
+ if defaults is None:
344
+ defaults = ()
345
+ first_default = len(args) - len(defaults)
346
+ for ix, name in enumerate(args):
347
+ if name not in all:
348
+ if ix >= first_default:
349
+ all[name] = defaults[ix - first_default]
350
+ else:
351
+ raise TypeError("Missing argument: %s" % name)
352
+ return all
353
+
354
+
355
+ def get_body(source):
356
+ ix = source.index(':')
357
+ if source[:5] == 'lambda':
358
+ return "return %s" % source[ix+1:]
359
+ else:
360
+ return source[ix+1:]
361
+
362
+
363
+ # Lots to be done here... It would be especially cool if compiled functions
364
+ # could invoke each other quickly.
365
+ class RuntimeCompiledFunction(object):
366
+
367
+ def __init__(self, f):
368
+ self._f = f
369
+ self._body = get_body(inspect.getsource(f))
370
+
371
+ def __call__(self, *args, **kwds):
372
+ all = getcallargs(self._f, *args, **kwds)
373
+ if IS_PY3:
374
+ return cython_inline(self._body, locals=self._f.__globals__, globals=self._f.__globals__, **all)
375
+ else:
376
+ return cython_inline(self._body, locals=self._f.func_globals, globals=self._f.func_globals, **all)
Cython/Build/IpythonMagic.py ADDED
@@ -0,0 +1,565 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # -*- coding: utf-8 -*-
2
+ """
3
+ =====================
4
+ Cython related magics
5
+ =====================
6
+
7
+ Magic command interface for interactive work with Cython
8
+
9
+ .. note::
10
+
11
+ The ``Cython`` package needs to be installed separately. It
12
+ can be obtained using ``easy_install`` or ``pip``.
13
+
14
+ Usage
15
+ =====
16
+
17
+ To enable the magics below, execute ``%load_ext cython``.
18
+
19
+ ``%%cython``
20
+
21
+ {CYTHON_DOC}
22
+
23
+ ``%%cython_inline``
24
+
25
+ {CYTHON_INLINE_DOC}
26
+
27
+ ``%%cython_pyximport``
28
+
29
+ {CYTHON_PYXIMPORT_DOC}
30
+
31
+ Author:
32
+ * Brian Granger
33
+
34
+ Code moved from IPython and adapted by:
35
+ * Martín Gaitán
36
+
37
+ Parts of this code were taken from Cython.inline.
38
+ """
39
+ #-----------------------------------------------------------------------------
40
+ # Copyright (C) 2010-2011, IPython Development Team.
41
+ #
42
+ # Distributed under the terms of the Modified BSD License.
43
+ #
44
+ # The full license is in the file ipython-COPYING.rst, distributed with this software.
45
+ #-----------------------------------------------------------------------------
46
+
47
+ from __future__ import absolute_import, print_function
48
+
49
+ import imp
50
+ import io
51
+ import os
52
+ import re
53
+ import sys
54
+ import time
55
+ import copy
56
+ import distutils.log
57
+ import textwrap
58
+
59
+ IO_ENCODING = sys.getfilesystemencoding()
60
+ IS_PY2 = sys.version_info[0] < 3
61
+
62
+ try:
63
+ reload
64
+ except NameError: # Python 3
65
+ from imp import reload
66
+
67
+ try:
68
+ import hashlib
69
+ except ImportError:
70
+ import md5 as hashlib
71
+
72
+ from distutils.core import Distribution, Extension
73
+ from distutils.command.build_ext import build_ext
74
+
75
+ from IPython.core import display
76
+ from IPython.core import magic_arguments
77
+ from IPython.core.magic import Magics, magics_class, cell_magic
78
+ try:
79
+ from IPython.paths import get_ipython_cache_dir
80
+ except ImportError:
81
+ # older IPython version
82
+ from IPython.utils.path import get_ipython_cache_dir
83
+ from IPython.utils.text import dedent
84
+
85
+ from ..Shadow import __version__ as cython_version
86
+ from ..Compiler.Errors import CompileError
87
+ from .Inline import cython_inline
88
+ from .Dependencies import cythonize
89
+
90
+
91
+ PGO_CONFIG = {
92
+ 'gcc': {
93
+ 'gen': ['-fprofile-generate', '-fprofile-dir={TEMPDIR}'],
94
+ 'use': ['-fprofile-use', '-fprofile-correction', '-fprofile-dir={TEMPDIR}'],
95
+ },
96
+ # blind copy from 'configure' script in CPython 3.7
97
+ 'icc': {
98
+ 'gen': ['-prof-gen'],
99
+ 'use': ['-prof-use'],
100
+ }
101
+ }
102
+ PGO_CONFIG['mingw32'] = PGO_CONFIG['gcc']
103
+
104
+
105
+ if IS_PY2:
106
+ def encode_fs(name):
107
+ return name if isinstance(name, bytes) else name.encode(IO_ENCODING)
108
+ else:
109
+ def encode_fs(name):
110
+ return name
111
+
112
+
113
+ @magics_class
114
+ class CythonMagics(Magics):
115
+
116
+ def __init__(self, shell):
117
+ super(CythonMagics, self).__init__(shell)
118
+ self._reloads = {}
119
+ self._code_cache = {}
120
+ self._pyximport_installed = False
121
+
122
+ def _import_all(self, module):
123
+ mdict = module.__dict__
124
+ if '__all__' in mdict:
125
+ keys = mdict['__all__']
126
+ else:
127
+ keys = [k for k in mdict if not k.startswith('_')]
128
+
129
+ for k in keys:
130
+ try:
131
+ self.shell.push({k: mdict[k]})
132
+ except KeyError:
133
+ msg = "'module' object has no attribute '%s'" % k
134
+ raise AttributeError(msg)
135
+
136
+ @cell_magic
137
+ def cython_inline(self, line, cell):
138
+ """Compile and run a Cython code cell using Cython.inline.
139
+
140
+ This magic simply passes the body of the cell to Cython.inline
141
+ and returns the result. If the variables `a` and `b` are defined
142
+ in the user's namespace, here is a simple example that returns
143
+ their sum::
144
+
145
+ %%cython_inline
146
+ return a+b
147
+
148
+ For most purposes, we recommend the usage of the `%%cython` magic.
149
+ """
150
+ locs = self.shell.user_global_ns
151
+ globs = self.shell.user_ns
152
+ return cython_inline(cell, locals=locs, globals=globs)
153
+
154
+ @cell_magic
155
+ def cython_pyximport(self, line, cell):
156
+ """Compile and import a Cython code cell using pyximport.
157
+
158
+ The contents of the cell are written to a `.pyx` file in the current
159
+ working directory, which is then imported using `pyximport`. This
160
+ magic requires a module name to be passed::
161
+
162
+ %%cython_pyximport modulename
163
+ def f(x):
164
+ return 2.0*x
165
+
166
+ The compiled module is then imported and all of its symbols are
167
+ injected into the user's namespace. For most purposes, we recommend
168
+ the usage of the `%%cython` magic.
169
+ """
170
+ module_name = line.strip()
171
+ if not module_name:
172
+ raise ValueError('module name must be given')
173
+ fname = module_name + '.pyx'
174
+ with io.open(fname, 'w', encoding='utf-8') as f:
175
+ f.write(cell)
176
+ if 'pyximport' not in sys.modules or not self._pyximport_installed:
177
+ import pyximport
178
+ pyximport.install()
179
+ self._pyximport_installed = True
180
+ if module_name in self._reloads:
181
+ module = self._reloads[module_name]
182
+ # Note: reloading extension modules is not actually supported
183
+ # (requires PEP-489 reinitialisation support).
184
+ # Don't know why this should ever have worked as it reads here.
185
+ # All we really need to do is to update the globals below.
186
+ #reload(module)
187
+ else:
188
+ __import__(module_name)
189
+ module = sys.modules[module_name]
190
+ self._reloads[module_name] = module
191
+ self._import_all(module)
192
+
193
+ @magic_arguments.magic_arguments()
194
+ @magic_arguments.argument(
195
+ '-a', '--annotate', action='store_true', default=False,
196
+ help="Produce a colorized HTML version of the source."
197
+ )
198
+ @magic_arguments.argument(
199
+ '-+', '--cplus', action='store_true', default=False,
200
+ help="Output a C++ rather than C file."
201
+ )
202
+ @magic_arguments.argument(
203
+ '-3', dest='language_level', action='store_const', const=3, default=None,
204
+ help="Select Python 3 syntax."
205
+ )
206
+ @magic_arguments.argument(
207
+ '-2', dest='language_level', action='store_const', const=2, default=None,
208
+ help="Select Python 2 syntax."
209
+ )
210
+ @magic_arguments.argument(
211
+ '-f', '--force', action='store_true', default=False,
212
+ help="Force the compilation of a new module, even if the source has been "
213
+ "previously compiled."
214
+ )
215
+ @magic_arguments.argument(
216
+ '-c', '--compile-args', action='append', default=[],
217
+ help="Extra flags to pass to compiler via the `extra_compile_args` "
218
+ "Extension flag (can be specified multiple times)."
219
+ )
220
+ @magic_arguments.argument(
221
+ '--link-args', action='append', default=[],
222
+ help="Extra flags to pass to linker via the `extra_link_args` "
223
+ "Extension flag (can be specified multiple times)."
224
+ )
225
+ @magic_arguments.argument(
226
+ '-l', '--lib', action='append', default=[],
227
+ help="Add a library to link the extension against (can be specified "
228
+ "multiple times)."
229
+ )
230
+ @magic_arguments.argument(
231
+ '-n', '--name',
232
+ help="Specify a name for the Cython module."
233
+ )
234
+ @magic_arguments.argument(
235
+ '-L', dest='library_dirs', metavar='dir', action='append', default=[],
236
+ help="Add a path to the list of library directories (can be specified "
237
+ "multiple times)."
238
+ )
239
+ @magic_arguments.argument(
240
+ '-I', '--include', action='append', default=[],
241
+ help="Add a path to the list of include directories (can be specified "
242
+ "multiple times)."
243
+ )
244
+ @magic_arguments.argument(
245
+ '-S', '--src', action='append', default=[],
246
+ help="Add a path to the list of src files (can be specified "
247
+ "multiple times)."
248
+ )
249
+ @magic_arguments.argument(
250
+ '--pgo', dest='pgo', action='store_true', default=False,
251
+ help=("Enable profile guided optimisation in the C compiler. "
252
+ "Compiles the cell twice and executes it in between to generate a runtime profile.")
253
+ )
254
+ @magic_arguments.argument(
255
+ '--verbose', dest='quiet', action='store_false', default=True,
256
+ help=("Print debug information like generated .c/.cpp file location "
257
+ "and exact gcc/g++ command invoked.")
258
+ )
259
+ @cell_magic
260
+ def cython(self, line, cell):
261
+ """Compile and import everything from a Cython code cell.
262
+
263
+ The contents of the cell are written to a `.pyx` file in the
264
+ directory `IPYTHONDIR/cython` using a filename with the hash of the
265
+ code. This file is then cythonized and compiled. The resulting module
266
+ is imported and all of its symbols are injected into the user's
267
+ namespace. The usage is similar to that of `%%cython_pyximport` but
268
+ you don't have to pass a module name::
269
+
270
+ %%cython
271
+ def f(x):
272
+ return 2.0*x
273
+
274
+ To compile OpenMP codes, pass the required `--compile-args`
275
+ and `--link-args`. For example with gcc::
276
+
277
+ %%cython --compile-args=-fopenmp --link-args=-fopenmp
278
+ ...
279
+
280
+ To enable profile guided optimisation, pass the ``--pgo`` option.
281
+ Note that the cell itself needs to take care of establishing a suitable
282
+ profile when executed. This can be done by implementing the functions to
283
+ optimise, and then calling them directly in the same cell on some realistic
284
+ training data like this::
285
+
286
+ %%cython --pgo
287
+ def critical_function(data):
288
+ for item in data:
289
+ ...
290
+
291
+ # execute function several times to build profile
292
+ from somewhere import some_typical_data
293
+ for _ in range(100):
294
+ critical_function(some_typical_data)
295
+
296
+ In Python 3.5 and later, you can distinguish between the profile and
297
+ non-profile runs as follows::
298
+
299
+ if "_pgo_" in __name__:
300
+ ... # execute critical code here
301
+ """
302
+ args = magic_arguments.parse_argstring(self.cython, line)
303
+ code = cell if cell.endswith('\n') else cell + '\n'
304
+ lib_dir = os.path.join(get_ipython_cache_dir(), 'cython')
305
+ key = (code, line, sys.version_info, sys.executable, cython_version)
306
+
307
+ if not os.path.exists(lib_dir):
308
+ os.makedirs(lib_dir)
309
+
310
+ if args.pgo:
311
+ key += ('pgo',)
312
+ if args.force:
313
+ # Force a new module name by adding the current time to the
314
+ # key which is hashed to determine the module name.
315
+ key += (time.time(),)
316
+
317
+ if args.name:
318
+ module_name = str(args.name) # no-op in Py3
319
+ else:
320
+ module_name = "_cython_magic_" + hashlib.md5(str(key).encode('utf-8')).hexdigest()
321
+ html_file = os.path.join(lib_dir, module_name + '.html')
322
+ module_path = os.path.join(lib_dir, module_name + self.so_ext)
323
+
324
+ have_module = os.path.isfile(module_path)
325
+ need_cythonize = args.pgo or not have_module
326
+
327
+ if args.annotate:
328
+ if not os.path.isfile(html_file):
329
+ need_cythonize = True
330
+
331
+ extension = None
332
+ if need_cythonize:
333
+ extensions = self._cythonize(module_name, code, lib_dir, args, quiet=args.quiet)
334
+ if extensions is None:
335
+ # Compilation failed and printed error message
336
+ return None
337
+ assert len(extensions) == 1
338
+ extension = extensions[0]
339
+ self._code_cache[key] = module_name
340
+
341
+ if args.pgo:
342
+ self._profile_pgo_wrapper(extension, lib_dir)
343
+
344
+ try:
345
+ self._build_extension(extension, lib_dir, pgo_step_name='use' if args.pgo else None,
346
+ quiet=args.quiet)
347
+ except distutils.errors.CompileError:
348
+ # Build failed and printed error message
349
+ return None
350
+
351
+ module = imp.load_dynamic(module_name, module_path)
352
+ self._import_all(module)
353
+
354
+ if args.annotate:
355
+ try:
356
+ with io.open(html_file, encoding='utf-8') as f:
357
+ annotated_html = f.read()
358
+ except IOError as e:
359
+ # File could not be opened. Most likely the user has a version
360
+ # of Cython before 0.15.1 (when `cythonize` learned the
361
+ # `force` keyword argument) and has already compiled this
362
+ # exact source without annotation.
363
+ print('Cython completed successfully but the annotated '
364
+ 'source could not be read.', file=sys.stderr)
365
+ print(e, file=sys.stderr)
366
+ else:
367
+ return display.HTML(self.clean_annotated_html(annotated_html))
368
+
369
+ def _profile_pgo_wrapper(self, extension, lib_dir):
370
+ """
371
+ Generate a .c file for a separate extension module that calls the
372
+ module init function of the original module. This makes sure that the
373
+ PGO profiler sees the correct .o file of the final module, but it still
374
+ allows us to import the module under a different name for profiling,
375
+ before recompiling it into the PGO optimised module. Overwriting and
376
+ reimporting the same shared library is not portable.
377
+ """
378
+ extension = copy.copy(extension) # shallow copy, do not modify sources in place!
379
+ module_name = extension.name
380
+ pgo_module_name = '_pgo_' + module_name
381
+ pgo_wrapper_c_file = os.path.join(lib_dir, pgo_module_name + '.c')
382
+ with io.open(pgo_wrapper_c_file, 'w', encoding='utf-8') as f:
383
+ f.write(textwrap.dedent(u"""
384
+ #include "Python.h"
385
+ #if PY_MAJOR_VERSION < 3
386
+ extern PyMODINIT_FUNC init%(module_name)s(void);
387
+ PyMODINIT_FUNC init%(pgo_module_name)s(void); /*proto*/
388
+ PyMODINIT_FUNC init%(pgo_module_name)s(void) {
389
+ PyObject *sys_modules;
390
+ init%(module_name)s(); if (PyErr_Occurred()) return;
391
+ sys_modules = PyImport_GetModuleDict(); /* borrowed, no exception, "never" fails */
392
+ if (sys_modules) {
393
+ PyObject *module = PyDict_GetItemString(sys_modules, "%(module_name)s"); if (!module) return;
394
+ PyDict_SetItemString(sys_modules, "%(pgo_module_name)s", module);
395
+ Py_DECREF(module);
396
+ }
397
+ }
398
+ #else
399
+ extern PyMODINIT_FUNC PyInit_%(module_name)s(void);
400
+ PyMODINIT_FUNC PyInit_%(pgo_module_name)s(void); /*proto*/
401
+ PyMODINIT_FUNC PyInit_%(pgo_module_name)s(void) {
402
+ return PyInit_%(module_name)s();
403
+ }
404
+ #endif
405
+ """ % {'module_name': module_name, 'pgo_module_name': pgo_module_name}))
406
+
407
+ extension.sources = extension.sources + [pgo_wrapper_c_file] # do not modify in place!
408
+ extension.name = pgo_module_name
409
+
410
+ self._build_extension(extension, lib_dir, pgo_step_name='gen')
411
+
412
+ # import and execute module code to generate profile
413
+ so_module_path = os.path.join(lib_dir, pgo_module_name + self.so_ext)
414
+ imp.load_dynamic(pgo_module_name, so_module_path)
415
+
416
+ def _cythonize(self, module_name, code, lib_dir, args, quiet=True):
417
+ pyx_file = os.path.join(lib_dir, module_name + '.pyx')
418
+ pyx_file = encode_fs(pyx_file)
419
+
420
+ c_include_dirs = args.include
421
+ c_src_files = list(map(str, args.src))
422
+ if 'numpy' in code:
423
+ import numpy
424
+ c_include_dirs.append(numpy.get_include())
425
+ with io.open(pyx_file, 'w', encoding='utf-8') as f:
426
+ f.write(code)
427
+ extension = Extension(
428
+ name=module_name,
429
+ sources=[pyx_file] + c_src_files,
430
+ include_dirs=c_include_dirs,
431
+ library_dirs=args.library_dirs,
432
+ extra_compile_args=args.compile_args,
433
+ extra_link_args=args.link_args,
434
+ libraries=args.lib,
435
+ language='c++' if args.cplus else 'c',
436
+ )
437
+ try:
438
+ opts = dict(
439
+ quiet=quiet,
440
+ annotate=args.annotate,
441
+ force=True,
442
+ )
443
+ if args.language_level is not None:
444
+ assert args.language_level in (2, 3)
445
+ opts['language_level'] = args.language_level
446
+ elif sys.version_info[0] >= 3:
447
+ opts['language_level'] = 3
448
+ return cythonize([extension], **opts)
449
+ except CompileError:
450
+ return None
451
+
452
+ def _build_extension(self, extension, lib_dir, temp_dir=None, pgo_step_name=None, quiet=True):
453
+ build_extension = self._get_build_extension(
454
+ extension, lib_dir=lib_dir, temp_dir=temp_dir, pgo_step_name=pgo_step_name)
455
+ old_threshold = None
456
+ try:
457
+ if not quiet:
458
+ old_threshold = distutils.log.set_threshold(distutils.log.DEBUG)
459
+ build_extension.run()
460
+ finally:
461
+ if not quiet and old_threshold is not None:
462
+ distutils.log.set_threshold(old_threshold)
463
+
464
+ def _add_pgo_flags(self, build_extension, step_name, temp_dir):
465
+ compiler_type = build_extension.compiler.compiler_type
466
+ if compiler_type == 'unix':
467
+ compiler_cmd = build_extension.compiler.compiler_so
468
+ # TODO: we could try to call "[cmd] --version" for better insights
469
+ if not compiler_cmd:
470
+ pass
471
+ elif 'clang' in compiler_cmd or 'clang' in compiler_cmd[0]:
472
+ compiler_type = 'clang'
473
+ elif 'icc' in compiler_cmd or 'icc' in compiler_cmd[0]:
474
+ compiler_type = 'icc'
475
+ elif 'gcc' in compiler_cmd or 'gcc' in compiler_cmd[0]:
476
+ compiler_type = 'gcc'
477
+ elif 'g++' in compiler_cmd or 'g++' in compiler_cmd[0]:
478
+ compiler_type = 'gcc'
479
+ config = PGO_CONFIG.get(compiler_type)
480
+ orig_flags = []
481
+ if config and step_name in config:
482
+ flags = [f.format(TEMPDIR=temp_dir) for f in config[step_name]]
483
+ for extension in build_extension.extensions:
484
+ orig_flags.append((extension.extra_compile_args, extension.extra_link_args))
485
+ extension.extra_compile_args = extension.extra_compile_args + flags
486
+ extension.extra_link_args = extension.extra_link_args + flags
487
+ else:
488
+ print("No PGO %s configuration known for C compiler type '%s'" % (step_name, compiler_type),
489
+ file=sys.stderr)
490
+ return orig_flags
491
+
492
+ @property
493
+ def so_ext(self):
494
+ """The extension suffix for compiled modules."""
495
+ try:
496
+ return self._so_ext
497
+ except AttributeError:
498
+ self._so_ext = self._get_build_extension().get_ext_filename('')
499
+ return self._so_ext
500
+
501
+ def _clear_distutils_mkpath_cache(self):
502
+ """clear distutils mkpath cache
503
+
504
+ prevents distutils from skipping re-creation of dirs that have been removed
505
+ """
506
+ try:
507
+ from distutils.dir_util import _path_created
508
+ except ImportError:
509
+ pass
510
+ else:
511
+ _path_created.clear()
512
+
513
+ def _get_build_extension(self, extension=None, lib_dir=None, temp_dir=None,
514
+ pgo_step_name=None, _build_ext=build_ext):
515
+ self._clear_distutils_mkpath_cache()
516
+ dist = Distribution()
517
+ config_files = dist.find_config_files()
518
+ try:
519
+ config_files.remove('setup.cfg')
520
+ except ValueError:
521
+ pass
522
+ dist.parse_config_files(config_files)
523
+
524
+ if not temp_dir:
525
+ temp_dir = lib_dir
526
+ add_pgo_flags = self._add_pgo_flags
527
+
528
+ if pgo_step_name:
529
+ base_build_ext = _build_ext
530
+ class _build_ext(_build_ext):
531
+ def build_extensions(self):
532
+ add_pgo_flags(self, pgo_step_name, temp_dir)
533
+ base_build_ext.build_extensions(self)
534
+
535
+ build_extension = _build_ext(dist)
536
+ build_extension.finalize_options()
537
+ if temp_dir:
538
+ temp_dir = encode_fs(temp_dir)
539
+ build_extension.build_temp = temp_dir
540
+ if lib_dir:
541
+ lib_dir = encode_fs(lib_dir)
542
+ build_extension.build_lib = lib_dir
543
+ if extension is not None:
544
+ build_extension.extensions = [extension]
545
+ return build_extension
546
+
547
+ @staticmethod
548
+ def clean_annotated_html(html):
549
+ """Clean up the annotated HTML source.
550
+
551
+ Strips the link to the generated C or C++ file, which we do not
552
+ present to the user.
553
+ """
554
+ r = re.compile('<p>Raw output: <a href="(.*)">(.*)</a>')
555
+ html = '\n'.join(l for l in html.splitlines() if not r.match(l))
556
+ return html
557
+
558
+ __doc__ = __doc__.format(
559
+ # rST doesn't see the -+ flag as part of an option list, so we
560
+ # hide it from the module-level docstring.
561
+ CYTHON_DOC=dedent(CythonMagics.cython.__doc__\
562
+ .replace('-+, --cplus', '--cplus ')),
563
+ CYTHON_INLINE_DOC=dedent(CythonMagics.cython_inline.__doc__),
564
+ CYTHON_PYXIMPORT_DOC=dedent(CythonMagics.cython_pyximport.__doc__),
565
+ )
Cython/Build/Tests/TestCyCache.py ADDED
@@ -0,0 +1,106 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import difflib
2
+ import glob
3
+ import gzip
4
+ import os
5
+ import tempfile
6
+
7
+ import Cython.Build.Dependencies
8
+ import Cython.Utils
9
+ from Cython.TestUtils import CythonTest
10
+
11
+
12
+ class TestCyCache(CythonTest):
13
+
14
+ def setUp(self):
15
+ CythonTest.setUp(self)
16
+ self.temp_dir = tempfile.mkdtemp(
17
+ prefix='cycache-test',
18
+ dir='TEST_TMP' if os.path.isdir('TEST_TMP') else None)
19
+ self.src_dir = tempfile.mkdtemp(prefix='src', dir=self.temp_dir)
20
+ self.cache_dir = tempfile.mkdtemp(prefix='cache', dir=self.temp_dir)
21
+
22
+ def cache_files(self, file_glob):
23
+ return glob.glob(os.path.join(self.cache_dir, file_glob))
24
+
25
+ def fresh_cythonize(self, *args, **kwargs):
26
+ Cython.Utils.clear_function_caches()
27
+ Cython.Build.Dependencies._dep_tree = None # discard method caches
28
+ Cython.Build.Dependencies.cythonize(*args, **kwargs)
29
+
30
+ def test_cycache_switch(self):
31
+ content1 = 'value = 1\n'
32
+ content2 = 'value = 2\n'
33
+ a_pyx = os.path.join(self.src_dir, 'a.pyx')
34
+ a_c = a_pyx[:-4] + '.c'
35
+
36
+ open(a_pyx, 'w').write(content1)
37
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
38
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
39
+ self.assertEqual(1, len(self.cache_files('a.c*')))
40
+ a_contents1 = open(a_c).read()
41
+ os.unlink(a_c)
42
+
43
+ open(a_pyx, 'w').write(content2)
44
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
45
+ a_contents2 = open(a_c).read()
46
+ os.unlink(a_c)
47
+
48
+ self.assertNotEqual(a_contents1, a_contents2, 'C file not changed!')
49
+ self.assertEqual(2, len(self.cache_files('a.c*')))
50
+
51
+ open(a_pyx, 'w').write(content1)
52
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
53
+ self.assertEqual(2, len(self.cache_files('a.c*')))
54
+ a_contents = open(a_c).read()
55
+ self.assertEqual(
56
+ a_contents, a_contents1,
57
+ msg='\n'.join(list(difflib.unified_diff(
58
+ a_contents.split('\n'), a_contents1.split('\n')))[:10]))
59
+
60
+ def test_cycache_uses_cache(self):
61
+ a_pyx = os.path.join(self.src_dir, 'a.pyx')
62
+ a_c = a_pyx[:-4] + '.c'
63
+ open(a_pyx, 'w').write('pass')
64
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
65
+ a_cache = os.path.join(self.cache_dir, os.listdir(self.cache_dir)[0])
66
+ gzip.GzipFile(a_cache, 'wb').write('fake stuff'.encode('ascii'))
67
+ os.unlink(a_c)
68
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
69
+ a_contents = open(a_c).read()
70
+ self.assertEqual(a_contents, 'fake stuff',
71
+ 'Unexpected contents: %s...' % a_contents[:100])
72
+
73
+ def test_multi_file_output(self):
74
+ a_pyx = os.path.join(self.src_dir, 'a.pyx')
75
+ a_c = a_pyx[:-4] + '.c'
76
+ a_h = a_pyx[:-4] + '.h'
77
+ a_api_h = a_pyx[:-4] + '_api.h'
78
+ open(a_pyx, 'w').write('cdef public api int foo(int x): return x\n')
79
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
80
+ expected = [a_c, a_h, a_api_h]
81
+ for output in expected:
82
+ self.assertTrue(os.path.exists(output), output)
83
+ os.unlink(output)
84
+ self.fresh_cythonize(a_pyx, cache=self.cache_dir)
85
+ for output in expected:
86
+ self.assertTrue(os.path.exists(output), output)
87
+
88
+ def test_options_invalidation(self):
89
+ hash_pyx = os.path.join(self.src_dir, 'options.pyx')
90
+ hash_c = hash_pyx[:-len('.pyx')] + '.c'
91
+
92
+ open(hash_pyx, 'w').write('pass')
93
+ self.fresh_cythonize(hash_pyx, cache=self.cache_dir, cplus=False)
94
+ self.assertEqual(1, len(self.cache_files('options.c*')))
95
+
96
+ os.unlink(hash_c)
97
+ self.fresh_cythonize(hash_pyx, cache=self.cache_dir, cplus=True)
98
+ self.assertEqual(2, len(self.cache_files('options.c*')))
99
+
100
+ os.unlink(hash_c)
101
+ self.fresh_cythonize(hash_pyx, cache=self.cache_dir, cplus=False, show_version=False)
102
+ self.assertEqual(2, len(self.cache_files('options.c*')))
103
+
104
+ os.unlink(hash_c)
105
+ self.fresh_cythonize(hash_pyx, cache=self.cache_dir, cplus=False, show_version=True)
106
+ self.assertEqual(2, len(self.cache_files('options.c*')))
Cython/Build/Tests/TestInline.py ADDED
@@ -0,0 +1,96 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import os, tempfile
2
+ from Cython.Shadow import inline
3
+ from Cython.Build.Inline import safe_type
4
+ from Cython.TestUtils import CythonTest
5
+
6
+ try:
7
+ import numpy
8
+ has_numpy = True
9
+ except:
10
+ has_numpy = False
11
+
12
+ test_kwds = dict(force=True, quiet=True)
13
+
14
+ global_value = 100
15
+
16
+ class TestInline(CythonTest):
17
+ def setUp(self):
18
+ CythonTest.setUp(self)
19
+ self.test_kwds = dict(test_kwds)
20
+ if os.path.isdir('TEST_TMP'):
21
+ lib_dir = os.path.join('TEST_TMP','inline')
22
+ else:
23
+ lib_dir = tempfile.mkdtemp(prefix='cython_inline_')
24
+ self.test_kwds['lib_dir'] = lib_dir
25
+
26
+ def test_simple(self):
27
+ self.assertEqual(inline("return 1+2", **self.test_kwds), 3)
28
+
29
+ def test_types(self):
30
+ self.assertEqual(inline("""
31
+ cimport cython
32
+ return cython.typeof(a), cython.typeof(b)
33
+ """, a=1.0, b=[], **self.test_kwds), ('double', 'list object'))
34
+
35
+ def test_locals(self):
36
+ a = 1
37
+ b = 2
38
+ self.assertEqual(inline("return a+b", **self.test_kwds), 3)
39
+
40
+ def test_globals(self):
41
+ self.assertEqual(inline("return global_value + 1", **self.test_kwds), global_value + 1)
42
+
43
+ def test_no_return(self):
44
+ self.assertEqual(inline("""
45
+ a = 1
46
+ cdef double b = 2
47
+ cdef c = []
48
+ """, **self.test_kwds), dict(a=1, b=2.0, c=[]))
49
+
50
+ def test_def_node(self):
51
+ foo = inline("def foo(x): return x * x", **self.test_kwds)['foo']
52
+ self.assertEqual(foo(7), 49)
53
+
54
+ def test_class_ref(self):
55
+ class Type(object):
56
+ pass
57
+ tp = inline("Type")['Type']
58
+ self.assertEqual(tp, Type)
59
+
60
+ def test_pure(self):
61
+ import cython as cy
62
+ b = inline("""
63
+ b = cy.declare(float, a)
64
+ c = cy.declare(cy.pointer(cy.float), &b)
65
+ return b
66
+ """, a=3, **self.test_kwds)
67
+ self.assertEqual(type(b), float)
68
+
69
+ def test_compiler_directives(self):
70
+ self.assertEqual(
71
+ inline('return sum(x)',
72
+ x=[1, 2, 3],
73
+ cython_compiler_directives={'boundscheck': False}),
74
+ 6
75
+ )
76
+
77
+ def test_lang_version(self):
78
+ # GH-3419. Caching for inline code didn't always respect compiler directives.
79
+ inline_divcode = "def f(int a, int b): return a/b"
80
+ self.assertEqual(
81
+ inline(inline_divcode, language_level=2)['f'](5,2),
82
+ 2
83
+ )
84
+ self.assertEqual(
85
+ inline(inline_divcode, language_level=3)['f'](5,2),
86
+ 2.5
87
+ )
88
+
89
+ if has_numpy:
90
+
91
+ def test_numpy(self):
92
+ import numpy
93
+ a = numpy.ndarray((10, 20))
94
+ a[0,0] = 10
95
+ self.assertEqual(safe_type(a), 'numpy.ndarray[numpy.float64_t, ndim=2]')
96
+ self.assertEqual(inline("return a[0,0]", a=a, **self.test_kwds), 10.0)
Cython/Build/Tests/TestIpythonMagic.py ADDED
@@ -0,0 +1,205 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # -*- coding: utf-8 -*-
2
+ # tag: ipython
3
+
4
+ """Tests for the Cython magics extension."""
5
+
6
+ from __future__ import absolute_import
7
+
8
+ import os
9
+ import sys
10
+ from contextlib import contextmanager
11
+ from Cython.Build import IpythonMagic
12
+ from Cython.TestUtils import CythonTest
13
+
14
+ try:
15
+ import IPython.testing.globalipapp
16
+ except ImportError:
17
+ # Disable tests and fake helpers for initialisation below.
18
+ def skip_if_not_installed(_):
19
+ return None
20
+ else:
21
+ def skip_if_not_installed(c):
22
+ return c
23
+
24
+ try:
25
+ # disable IPython history thread before it gets started to avoid having to clean it up
26
+ from IPython.core.history import HistoryManager
27
+ HistoryManager.enabled = False
28
+ except ImportError:
29
+ pass
30
+
31
+ code = u"""\
32
+ def f(x):
33
+ return 2*x
34
+ """
35
+
36
+ cython3_code = u"""\
37
+ def f(int x):
38
+ return 2 / x
39
+
40
+ def call(x):
41
+ return f(*(x,))
42
+ """
43
+
44
+ pgo_cython3_code = cython3_code + u"""\
45
+ def main():
46
+ for _ in range(100): call(5)
47
+ main()
48
+ """
49
+
50
+
51
+ if sys.platform == 'win32':
52
+ # not using IPython's decorators here because they depend on "nose"
53
+ try:
54
+ from unittest import skip as skip_win32
55
+ except ImportError:
56
+ # poor dev's silent @unittest.skip()
57
+ def skip_win32(dummy):
58
+ def _skip_win32(func):
59
+ return None
60
+ return _skip_win32
61
+ else:
62
+ def skip_win32(dummy):
63
+ def _skip_win32(func):
64
+ def wrapper(*args, **kwargs):
65
+ func(*args, **kwargs)
66
+ return wrapper
67
+ return _skip_win32
68
+
69
+
70
+ @skip_if_not_installed
71
+ class TestIPythonMagic(CythonTest):
72
+
73
+ @classmethod
74
+ def setUpClass(cls):
75
+ CythonTest.setUpClass()
76
+ cls._ip = IPython.testing.globalipapp.get_ipython()
77
+
78
+ def setUp(self):
79
+ CythonTest.setUp(self)
80
+ self._ip.extension_manager.load_extension('cython')
81
+
82
+ def test_cython_inline(self):
83
+ ip = self._ip
84
+ ip.ex('a=10; b=20')
85
+ result = ip.run_cell_magic('cython_inline', '', 'return a+b')
86
+ self.assertEqual(result, 30)
87
+
88
+ @skip_win32('Skip on Windows')
89
+ def test_cython_pyximport(self):
90
+ ip = self._ip
91
+ module_name = '_test_cython_pyximport'
92
+ ip.run_cell_magic('cython_pyximport', module_name, code)
93
+ ip.ex('g = f(10)')
94
+ self.assertEqual(ip.user_ns['g'], 20.0)
95
+ ip.run_cell_magic('cython_pyximport', module_name, code)
96
+ ip.ex('h = f(-10)')
97
+ self.assertEqual(ip.user_ns['h'], -20.0)
98
+ try:
99
+ os.remove(module_name + '.pyx')
100
+ except OSError:
101
+ pass
102
+
103
+ def test_cython(self):
104
+ ip = self._ip
105
+ ip.run_cell_magic('cython', '', code)
106
+ ip.ex('g = f(10)')
107
+ self.assertEqual(ip.user_ns['g'], 20.0)
108
+
109
+ def test_cython_name(self):
110
+ # The Cython module named 'mymodule' defines the function f.
111
+ ip = self._ip
112
+ ip.run_cell_magic('cython', '--name=mymodule', code)
113
+ # This module can now be imported in the interactive namespace.
114
+ ip.ex('import mymodule; g = mymodule.f(10)')
115
+ self.assertEqual(ip.user_ns['g'], 20.0)
116
+
117
+ def test_cython_language_level(self):
118
+ # The Cython cell defines the functions f() and call().
119
+ ip = self._ip
120
+ ip.run_cell_magic('cython', '', cython3_code)
121
+ ip.ex('g = f(10); h = call(10)')
122
+ if sys.version_info[0] < 3:
123
+ self.assertEqual(ip.user_ns['g'], 2 // 10)
124
+ self.assertEqual(ip.user_ns['h'], 2 // 10)
125
+ else:
126
+ self.assertEqual(ip.user_ns['g'], 2.0 / 10.0)
127
+ self.assertEqual(ip.user_ns['h'], 2.0 / 10.0)
128
+
129
+ def test_cython3(self):
130
+ # The Cython cell defines the functions f() and call().
131
+ ip = self._ip
132
+ ip.run_cell_magic('cython', '-3', cython3_code)
133
+ ip.ex('g = f(10); h = call(10)')
134
+ self.assertEqual(ip.user_ns['g'], 2.0 / 10.0)
135
+ self.assertEqual(ip.user_ns['h'], 2.0 / 10.0)
136
+
137
+ def test_cython2(self):
138
+ # The Cython cell defines the functions f() and call().
139
+ ip = self._ip
140
+ ip.run_cell_magic('cython', '-2', cython3_code)
141
+ ip.ex('g = f(10); h = call(10)')
142
+ self.assertEqual(ip.user_ns['g'], 2 // 10)
143
+ self.assertEqual(ip.user_ns['h'], 2 // 10)
144
+
145
+ @skip_win32('Skip on Windows')
146
+ def test_cython3_pgo(self):
147
+ # The Cython cell defines the functions f() and call().
148
+ ip = self._ip
149
+ ip.run_cell_magic('cython', '-3 --pgo', pgo_cython3_code)
150
+ ip.ex('g = f(10); h = call(10); main()')
151
+ self.assertEqual(ip.user_ns['g'], 2.0 / 10.0)
152
+ self.assertEqual(ip.user_ns['h'], 2.0 / 10.0)
153
+
154
+ @skip_win32('Skip on Windows')
155
+ def test_extlibs(self):
156
+ ip = self._ip
157
+ code = u"""
158
+ from libc.math cimport sin
159
+ x = sin(0.0)
160
+ """
161
+ ip.user_ns['x'] = 1
162
+ ip.run_cell_magic('cython', '-l m', code)
163
+ self.assertEqual(ip.user_ns['x'], 0)
164
+
165
+
166
+ def test_cython_verbose(self):
167
+ ip = self._ip
168
+ ip.run_cell_magic('cython', '--verbose', code)
169
+ ip.ex('g = f(10)')
170
+ self.assertEqual(ip.user_ns['g'], 20.0)
171
+
172
+ def test_cython_verbose_thresholds(self):
173
+ @contextmanager
174
+ def mock_distutils():
175
+ class MockLog:
176
+ DEBUG = 1
177
+ INFO = 2
178
+ thresholds = [INFO]
179
+
180
+ def set_threshold(self, val):
181
+ self.thresholds.append(val)
182
+ return self.thresholds[-2]
183
+
184
+
185
+ new_log = MockLog()
186
+ old_log = IpythonMagic.distutils.log
187
+ try:
188
+ IpythonMagic.distutils.log = new_log
189
+ yield new_log
190
+ finally:
191
+ IpythonMagic.distutils.log = old_log
192
+
193
+ ip = self._ip
194
+ with mock_distutils() as verbose_log:
195
+ ip.run_cell_magic('cython', '--verbose', code)
196
+ ip.ex('g = f(10)')
197
+ self.assertEqual(ip.user_ns['g'], 20.0)
198
+ self.assertEqual([verbose_log.INFO, verbose_log.DEBUG, verbose_log.INFO],
199
+ verbose_log.thresholds)
200
+
201
+ with mock_distutils() as normal_log:
202
+ ip.run_cell_magic('cython', '', code)
203
+ ip.ex('g = f(10)')
204
+ self.assertEqual(ip.user_ns['g'], 20.0)
205
+ self.assertEqual([normal_log.INFO], normal_log.thresholds)
Cython/Build/Tests/TestStripLiterals.py ADDED
@@ -0,0 +1,57 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from Cython.Build.Dependencies import strip_string_literals
2
+
3
+ from Cython.TestUtils import CythonTest
4
+
5
+ class TestStripLiterals(CythonTest):
6
+
7
+ def t(self, before, expected):
8
+ actual, literals = strip_string_literals(before, prefix="_L")
9
+ self.assertEqual(expected, actual)
10
+ for key, value in literals.items():
11
+ actual = actual.replace(key, value)
12
+ self.assertEqual(before, actual)
13
+
14
+ def test_empty(self):
15
+ self.t("", "")
16
+
17
+ def test_single_quote(self):
18
+ self.t("'x'", "'_L1_'")
19
+
20
+ def test_double_quote(self):
21
+ self.t('"x"', '"_L1_"')
22
+
23
+ def test_nested_quotes(self):
24
+ self.t(""" '"' "'" """, """ '_L1_' "_L2_" """)
25
+
26
+ def test_triple_quote(self):
27
+ self.t(" '''a\n''' ", " '''_L1_''' ")
28
+
29
+ def test_backslash(self):
30
+ self.t(r"'a\'b'", "'_L1_'")
31
+ self.t(r"'a\\'", "'_L1_'")
32
+ self.t(r"'a\\\'b'", "'_L1_'")
33
+
34
+ def test_unicode(self):
35
+ self.t("u'abc'", "u'_L1_'")
36
+
37
+ def test_raw(self):
38
+ self.t(r"r'abc\\'", "r'_L1_'")
39
+
40
+ def test_raw_unicode(self):
41
+ self.t(r"ru'abc\\'", "ru'_L1_'")
42
+
43
+ def test_comment(self):
44
+ self.t("abc # foo", "abc #_L1_")
45
+
46
+ def test_comment_and_quote(self):
47
+ self.t("abc # 'x'", "abc #_L1_")
48
+ self.t("'abc#'", "'_L1_'")
49
+
50
+ def test_include(self):
51
+ self.t("include 'a.pxi' # something here",
52
+ "include '_L1_' #_L2_")
53
+
54
+ def test_extern(self):
55
+ self.t("cdef extern from 'a.h': # comment",
56
+ "cdef extern from '_L1_': #_L2_")
57
+
Cython/Build/Tests/__init__.py ADDED
@@ -0,0 +1 @@
 
 
1
+ # empty file
Cython/Build/Tests/__pycache__/TestCyCache.cpython-39.pyc ADDED
Binary file (3.83 kB). View file
 
Cython/Build/Tests/__pycache__/TestInline.cpython-39.pyc ADDED
Binary file (4.04 kB). View file
 
Cython/Build/Tests/__pycache__/TestIpythonMagic.cpython-39.pyc ADDED
Binary file (6.71 kB). View file
 
Cython/Build/Tests/__pycache__/TestStripLiterals.cpython-39.pyc ADDED
Binary file (3.08 kB). View file
 
Cython/Build/Tests/__pycache__/__init__.cpython-39.pyc ADDED
Binary file (139 Bytes). View file
 
Cython/Build/__init__.py ADDED
@@ -0,0 +1,2 @@
 
 
 
1
+ from .Dependencies import cythonize
2
+ from .Distutils import build_ext
Cython/Build/__pycache__/BuildExecutable.cpython-39.pyc ADDED
Binary file (4.4 kB). View file
 
Cython/Build/__pycache__/Cythonize.cpython-39.pyc ADDED
Binary file (6.63 kB). View file
 
Cython/Build/__pycache__/Dependencies.cpython-39.pyc ADDED
Binary file (35.5 kB). View file
 
Cython/Build/__pycache__/Distutils.cpython-39.pyc ADDED
Binary file (194 Bytes). View file
 
Cython/Build/__pycache__/Inline.cpython-39.pyc ADDED
Binary file (10.6 kB). View file
 
Cython/Build/__pycache__/IpythonMagic.cpython-39.pyc ADDED
Binary file (16.8 kB). View file
 
Cython/Build/__pycache__/__init__.cpython-39.pyc ADDED
Binary file (222 Bytes). View file
 
Cython/CodeWriter.py ADDED
@@ -0,0 +1,816 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """
2
+ Serializes a Cython code tree to Cython code. This is primarily useful for
3
+ debugging and testing purposes.
4
+
5
+ The output is in a strict format, no whitespace or comments from the input
6
+ is preserved (and it could not be as it is not present in the code tree).
7
+ """
8
+
9
+ from __future__ import absolute_import, print_function
10
+
11
+ from .Compiler.Visitor import TreeVisitor
12
+ from .Compiler.ExprNodes import *
13
+
14
+
15
+ class LinesResult(object):
16
+ def __init__(self):
17
+ self.lines = []
18
+ self.s = u""
19
+
20
+ def put(self, s):
21
+ self.s += s
22
+
23
+ def newline(self):
24
+ self.lines.append(self.s)
25
+ self.s = u""
26
+
27
+ def putline(self, s):
28
+ self.put(s)
29
+ self.newline()
30
+
31
+ class DeclarationWriter(TreeVisitor):
32
+
33
+ indent_string = u" "
34
+
35
+ def __init__(self, result=None):
36
+ super(DeclarationWriter, self).__init__()
37
+ if result is None:
38
+ result = LinesResult()
39
+ self.result = result
40
+ self.numindents = 0
41
+ self.tempnames = {}
42
+ self.tempblockindex = 0
43
+
44
+ def write(self, tree):
45
+ self.visit(tree)
46
+ return self.result
47
+
48
+ def indent(self):
49
+ self.numindents += 1
50
+
51
+ def dedent(self):
52
+ self.numindents -= 1
53
+
54
+ def startline(self, s=u""):
55
+ self.result.put(self.indent_string * self.numindents + s)
56
+
57
+ def put(self, s):
58
+ self.result.put(s)
59
+
60
+ def putline(self, s):
61
+ self.result.putline(self.indent_string * self.numindents + s)
62
+
63
+ def endline(self, s=u""):
64
+ self.result.putline(s)
65
+
66
+ def line(self, s):
67
+ self.startline(s)
68
+ self.endline()
69
+
70
+ def comma_separated_list(self, items, output_rhs=False):
71
+ if len(items) > 0:
72
+ for item in items[:-1]:
73
+ self.visit(item)
74
+ if output_rhs and item.default is not None:
75
+ self.put(u" = ")
76
+ self.visit(item.default)
77
+ self.put(u", ")
78
+ self.visit(items[-1])
79
+
80
+ def visit_Node(self, node):
81
+ raise AssertionError("Node not handled by serializer: %r" % node)
82
+
83
+ def visit_ModuleNode(self, node):
84
+ self.visitchildren(node)
85
+
86
+ def visit_StatListNode(self, node):
87
+ self.visitchildren(node)
88
+
89
+ def visit_CDefExternNode(self, node):
90
+ if node.include_file is None:
91
+ file = u'*'
92
+ else:
93
+ file = u'"%s"' % node.include_file
94
+ self.putline(u"cdef extern from %s:" % file)
95
+ self.indent()
96
+ self.visit(node.body)
97
+ self.dedent()
98
+
99
+ def visit_CPtrDeclaratorNode(self, node):
100
+ self.put('*')
101
+ self.visit(node.base)
102
+
103
+ def visit_CReferenceDeclaratorNode(self, node):
104
+ self.put('&')
105
+ self.visit(node.base)
106
+
107
+ def visit_CArrayDeclaratorNode(self, node):
108
+ self.visit(node.base)
109
+ self.put(u'[')
110
+ if node.dimension is not None:
111
+ self.visit(node.dimension)
112
+ self.put(u']')
113
+
114
+ def visit_CArrayDeclaratorNode(self, node):
115
+ self.visit(node.base)
116
+ self.put(u'[')
117
+ if node.dimension is not None:
118
+ self.visit(node.dimension)
119
+ self.put(u']')
120
+
121
+ def visit_CFuncDeclaratorNode(self, node):
122
+ # TODO: except, gil, etc.
123
+ self.visit(node.base)
124
+ self.put(u'(')
125
+ self.comma_separated_list(node.args)
126
+ self.endline(u')')
127
+
128
+ def visit_CNameDeclaratorNode(self, node):
129
+ self.put(node.name)
130
+
131
+ def visit_CSimpleBaseTypeNode(self, node):
132
+ # See Parsing.p_sign_and_longness
133
+ if node.is_basic_c_type:
134
+ self.put(("unsigned ", "", "signed ")[node.signed])
135
+ if node.longness < 0:
136
+ self.put("short " * -node.longness)
137
+ elif node.longness > 0:
138
+ self.put("long " * node.longness)
139
+ self.put(node.name)
140
+
141
+ def visit_CComplexBaseTypeNode(self, node):
142
+ self.put(u'(')
143
+ self.visit(node.base_type)
144
+ self.visit(node.declarator)
145
+ self.put(u')')
146
+
147
+ def visit_CNestedBaseTypeNode(self, node):
148
+ self.visit(node.base_type)
149
+ self.put(u'.')
150
+ self.put(node.name)
151
+
152
+ def visit_TemplatedTypeNode(self, node):
153
+ self.visit(node.base_type_node)
154
+ self.put(u'[')
155
+ self.comma_separated_list(node.positional_args + node.keyword_args.key_value_pairs)
156
+ self.put(u']')
157
+
158
+ def visit_CVarDefNode(self, node):
159
+ self.startline(u"cdef ")
160
+ self.visit(node.base_type)
161
+ self.put(u" ")
162
+ self.comma_separated_list(node.declarators, output_rhs=True)
163
+ self.endline()
164
+
165
+ def visit_container_node(self, node, decl, extras, attributes):
166
+ # TODO: visibility
167
+ self.startline(decl)
168
+ if node.name:
169
+ self.put(u' ')
170
+ self.put(node.name)
171
+ if node.cname is not None:
172
+ self.put(u' "%s"' % node.cname)
173
+ if extras:
174
+ self.put(extras)
175
+ self.endline(':')
176
+ self.indent()
177
+ if not attributes:
178
+ self.putline('pass')
179
+ else:
180
+ for attribute in attributes:
181
+ self.visit(attribute)
182
+ self.dedent()
183
+
184
+ def visit_CStructOrUnionDefNode(self, node):
185
+ if node.typedef_flag:
186
+ decl = u'ctypedef '
187
+ else:
188
+ decl = u'cdef '
189
+ if node.visibility == 'public':
190
+ decl += u'public '
191
+ if node.packed:
192
+ decl += u'packed '
193
+ decl += node.kind
194
+ self.visit_container_node(node, decl, None, node.attributes)
195
+
196
+ def visit_CppClassNode(self, node):
197
+ extras = ""
198
+ if node.templates:
199
+ extras = u"[%s]" % ", ".join(node.templates)
200
+ if node.base_classes:
201
+ extras += "(%s)" % ", ".join(node.base_classes)
202
+ self.visit_container_node(node, u"cdef cppclass", extras, node.attributes)
203
+
204
+ def visit_CEnumDefNode(self, node):
205
+ self.visit_container_node(node, u"cdef enum", None, node.items)
206
+
207
+ def visit_CEnumDefItemNode(self, node):
208
+ self.startline(node.name)
209
+ if node.cname:
210
+ self.put(u' "%s"' % node.cname)
211
+ if node.value:
212
+ self.put(u" = ")
213
+ self.visit(node.value)
214
+ self.endline()
215
+
216
+ def visit_CClassDefNode(self, node):
217
+ assert not node.module_name
218
+ if node.decorators:
219
+ for decorator in node.decorators:
220
+ self.visit(decorator)
221
+ self.startline(u"cdef class ")
222
+ self.put(node.class_name)
223
+ if node.base_class_name:
224
+ self.put(u"(")
225
+ if node.base_class_module:
226
+ self.put(node.base_class_module)
227
+ self.put(u".")
228
+ self.put(node.base_class_name)
229
+ self.put(u")")
230
+ self.endline(u":")
231
+ self.indent()
232
+ self.visit(node.body)
233
+ self.dedent()
234
+
235
+ def visit_CTypeDefNode(self, node):
236
+ self.startline(u"ctypedef ")
237
+ self.visit(node.base_type)
238
+ self.put(u" ")
239
+ self.visit(node.declarator)
240
+ self.endline()
241
+
242
+ def visit_FuncDefNode(self, node):
243
+ self.startline(u"def %s(" % node.name)
244
+ self.comma_separated_list(node.args)
245
+ self.endline(u"):")
246
+ self.indent()
247
+ self.visit(node.body)
248
+ self.dedent()
249
+
250
+ def visit_CArgDeclNode(self, node):
251
+ if node.base_type.name is not None:
252
+ self.visit(node.base_type)
253
+ self.put(u" ")
254
+ self.visit(node.declarator)
255
+ if node.default is not None:
256
+ self.put(u" = ")
257
+ self.visit(node.default)
258
+
259
+ def visit_CImportStatNode(self, node):
260
+ self.startline(u"cimport ")
261
+ self.put(node.module_name)
262
+ if node.as_name:
263
+ self.put(u" as ")
264
+ self.put(node.as_name)
265
+ self.endline()
266
+
267
+ def visit_FromCImportStatNode(self, node):
268
+ self.startline(u"from ")
269
+ self.put(node.module_name)
270
+ self.put(u" cimport ")
271
+ first = True
272
+ for pos, name, as_name, kind in node.imported_names:
273
+ assert kind is None
274
+ if first:
275
+ first = False
276
+ else:
277
+ self.put(u", ")
278
+ self.put(name)
279
+ if as_name:
280
+ self.put(u" as ")
281
+ self.put(as_name)
282
+ self.endline()
283
+
284
+ def visit_NameNode(self, node):
285
+ self.put(node.name)
286
+
287
+ def visit_IntNode(self, node):
288
+ self.put(node.value)
289
+
290
+ def visit_NoneNode(self, node):
291
+ self.put(u"None")
292
+
293
+ def visit_NotNode(self, node):
294
+ self.put(u"(not ")
295
+ self.visit(node.operand)
296
+ self.put(u")")
297
+
298
+ def visit_DecoratorNode(self, node):
299
+ self.startline("@")
300
+ self.visit(node.decorator)
301
+ self.endline()
302
+
303
+ def visit_BinopNode(self, node):
304
+ self.visit(node.operand1)
305
+ self.put(u" %s " % node.operator)
306
+ self.visit(node.operand2)
307
+
308
+ def visit_AttributeNode(self, node):
309
+ self.visit(node.obj)
310
+ self.put(u".%s" % node.attribute)
311
+
312
+ def visit_BoolNode(self, node):
313
+ self.put(str(node.value))
314
+
315
+ # FIXME: represent string nodes correctly
316
+ def visit_StringNode(self, node):
317
+ value = node.value
318
+ if value.encoding is not None:
319
+ value = value.encode(value.encoding)
320
+ self.put(repr(value))
321
+
322
+ def visit_PassStatNode(self, node):
323
+ self.startline(u"pass")
324
+ self.endline()
325
+
326
+ class CodeWriter(DeclarationWriter):
327
+
328
+ def visit_SingleAssignmentNode(self, node):
329
+ self.startline()
330
+ self.visit(node.lhs)
331
+ self.put(u" = ")
332
+ self.visit(node.rhs)
333
+ self.endline()
334
+
335
+ def visit_CascadedAssignmentNode(self, node):
336
+ self.startline()
337
+ for lhs in node.lhs_list:
338
+ self.visit(lhs)
339
+ self.put(u" = ")
340
+ self.visit(node.rhs)
341
+ self.endline()
342
+
343
+ def visit_PrintStatNode(self, node):
344
+ self.startline(u"print ")
345
+ self.comma_separated_list(node.arg_tuple.args)
346
+ if not node.append_newline:
347
+ self.put(u",")
348
+ self.endline()
349
+
350
+ def visit_ForInStatNode(self, node):
351
+ self.startline(u"for ")
352
+ self.visit(node.target)
353
+ self.put(u" in ")
354
+ self.visit(node.iterator.sequence)
355
+ self.endline(u":")
356
+ self.indent()
357
+ self.visit(node.body)
358
+ self.dedent()
359
+ if node.else_clause is not None:
360
+ self.line(u"else:")
361
+ self.indent()
362
+ self.visit(node.else_clause)
363
+ self.dedent()
364
+
365
+ def visit_IfStatNode(self, node):
366
+ # The IfClauseNode is handled directly without a separate match
367
+ # for clariy.
368
+ self.startline(u"if ")
369
+ self.visit(node.if_clauses[0].condition)
370
+ self.endline(":")
371
+ self.indent()
372
+ self.visit(node.if_clauses[0].body)
373
+ self.dedent()
374
+ for clause in node.if_clauses[1:]:
375
+ self.startline("elif ")
376
+ self.visit(clause.condition)
377
+ self.endline(":")
378
+ self.indent()
379
+ self.visit(clause.body)
380
+ self.dedent()
381
+ if node.else_clause is not None:
382
+ self.line("else:")
383
+ self.indent()
384
+ self.visit(node.else_clause)
385
+ self.dedent()
386
+
387
+ def visit_SequenceNode(self, node):
388
+ self.comma_separated_list(node.args) # Might need to discover whether we need () around tuples...hmm...
389
+
390
+ def visit_SimpleCallNode(self, node):
391
+ self.visit(node.function)
392
+ self.put(u"(")
393
+ self.comma_separated_list(node.args)
394
+ self.put(")")
395
+
396
+ def visit_GeneralCallNode(self, node):
397
+ self.visit(node.function)
398
+ self.put(u"(")
399
+ posarg = node.positional_args
400
+ if isinstance(posarg, AsTupleNode):
401
+ self.visit(posarg.arg)
402
+ else:
403
+ self.comma_separated_list(posarg.args) # TupleNode.args
404
+ if node.keyword_args:
405
+ if isinstance(node.keyword_args, DictNode):
406
+ for i, (name, value) in enumerate(node.keyword_args.key_value_pairs):
407
+ if i > 0:
408
+ self.put(', ')
409
+ self.visit(name)
410
+ self.put('=')
411
+ self.visit(value)
412
+ else:
413
+ raise Exception("Not implemented yet")
414
+ self.put(u")")
415
+
416
+ def visit_ExprStatNode(self, node):
417
+ self.startline()
418
+ self.visit(node.expr)
419
+ self.endline()
420
+
421
+ def visit_InPlaceAssignmentNode(self, node):
422
+ self.startline()
423
+ self.visit(node.lhs)
424
+ self.put(u" %s= " % node.operator)
425
+ self.visit(node.rhs)
426
+ self.endline()
427
+
428
+ def visit_WithStatNode(self, node):
429
+ self.startline()
430
+ self.put(u"with ")
431
+ self.visit(node.manager)
432
+ if node.target is not None:
433
+ self.put(u" as ")
434
+ self.visit(node.target)
435
+ self.endline(u":")
436
+ self.indent()
437
+ self.visit(node.body)
438
+ self.dedent()
439
+
440
+ def visit_TryFinallyStatNode(self, node):
441
+ self.line(u"try:")
442
+ self.indent()
443
+ self.visit(node.body)
444
+ self.dedent()
445
+ self.line(u"finally:")
446
+ self.indent()
447
+ self.visit(node.finally_clause)
448
+ self.dedent()
449
+
450
+ def visit_TryExceptStatNode(self, node):
451
+ self.line(u"try:")
452
+ self.indent()
453
+ self.visit(node.body)
454
+ self.dedent()
455
+ for x in node.except_clauses:
456
+ self.visit(x)
457
+ if node.else_clause is not None:
458
+ self.visit(node.else_clause)
459
+
460
+ def visit_ExceptClauseNode(self, node):
461
+ self.startline(u"except")
462
+ if node.pattern is not None:
463
+ self.put(u" ")
464
+ self.visit(node.pattern)
465
+ if node.target is not None:
466
+ self.put(u", ")
467
+ self.visit(node.target)
468
+ self.endline(":")
469
+ self.indent()
470
+ self.visit(node.body)
471
+ self.dedent()
472
+
473
+ def visit_ReturnStatNode(self, node):
474
+ self.startline("return ")
475
+ self.visit(node.value)
476
+ self.endline()
477
+
478
+ def visit_ReraiseStatNode(self, node):
479
+ self.line("raise")
480
+
481
+ def visit_ImportNode(self, node):
482
+ self.put(u"(import %s)" % node.module_name.value)
483
+
484
+ def visit_TempsBlockNode(self, node):
485
+ """
486
+ Temporaries are output like $1_1', where the first number is
487
+ an index of the TempsBlockNode and the second number is an index
488
+ of the temporary which that block allocates.
489
+ """
490
+ idx = 0
491
+ for handle in node.temps:
492
+ self.tempnames[handle] = "$%d_%d" % (self.tempblockindex, idx)
493
+ idx += 1
494
+ self.tempblockindex += 1
495
+ self.visit(node.body)
496
+
497
+ def visit_TempRefNode(self, node):
498
+ self.put(self.tempnames[node.handle])
499
+
500
+
501
+ class PxdWriter(DeclarationWriter):
502
+ def __call__(self, node):
503
+ print(u'\n'.join(self.write(node).lines))
504
+ return node
505
+
506
+ def visit_CFuncDefNode(self, node):
507
+ if 'inline' in node.modifiers:
508
+ return
509
+ if node.overridable:
510
+ self.startline(u'cpdef ')
511
+ else:
512
+ self.startline(u'cdef ')
513
+ if node.visibility != 'private':
514
+ self.put(node.visibility)
515
+ self.put(u' ')
516
+ if node.api:
517
+ self.put(u'api ')
518
+ self.visit(node.declarator)
519
+
520
+ def visit_StatNode(self, node):
521
+ pass
522
+
523
+
524
+ class ExpressionWriter(TreeVisitor):
525
+
526
+ def __init__(self, result=None):
527
+ super(ExpressionWriter, self).__init__()
528
+ if result is None:
529
+ result = u""
530
+ self.result = result
531
+ self.precedence = [0]
532
+
533
+ def write(self, tree):
534
+ self.visit(tree)
535
+ return self.result
536
+
537
+ def put(self, s):
538
+ self.result += s
539
+
540
+ def remove(self, s):
541
+ if self.result.endswith(s):
542
+ self.result = self.result[:-len(s)]
543
+
544
+ def comma_separated_list(self, items):
545
+ if len(items) > 0:
546
+ for item in items[:-1]:
547
+ self.visit(item)
548
+ self.put(u", ")
549
+ self.visit(items[-1])
550
+
551
+ def visit_Node(self, node):
552
+ raise AssertionError("Node not handled by serializer: %r" % node)
553
+
554
+ def visit_NameNode(self, node):
555
+ self.put(node.name)
556
+
557
+ def visit_NoneNode(self, node):
558
+ self.put(u"None")
559
+
560
+ def visit_EllipsisNode(self, node):
561
+ self.put(u"...")
562
+
563
+ def visit_BoolNode(self, node):
564
+ self.put(str(node.value))
565
+
566
+ def visit_ConstNode(self, node):
567
+ self.put(str(node.value))
568
+
569
+ def visit_ImagNode(self, node):
570
+ self.put(node.value)
571
+ self.put(u"j")
572
+
573
+ def emit_string(self, node, prefix=u""):
574
+ repr_val = repr(node.value)
575
+ if repr_val[0] in 'ub':
576
+ repr_val = repr_val[1:]
577
+ self.put(u"%s%s" % (prefix, repr_val))
578
+
579
+ def visit_BytesNode(self, node):
580
+ self.emit_string(node, u"b")
581
+
582
+ def visit_StringNode(self, node):
583
+ self.emit_string(node)
584
+
585
+ def visit_UnicodeNode(self, node):
586
+ self.emit_string(node, u"u")
587
+
588
+ def emit_sequence(self, node, parens=(u"", u"")):
589
+ open_paren, close_paren = parens
590
+ items = node.subexpr_nodes()
591
+ self.put(open_paren)
592
+ self.comma_separated_list(items)
593
+ self.put(close_paren)
594
+
595
+ def visit_ListNode(self, node):
596
+ self.emit_sequence(node, u"[]")
597
+
598
+ def visit_TupleNode(self, node):
599
+ self.emit_sequence(node, u"()")
600
+
601
+ def visit_SetNode(self, node):
602
+ if len(node.subexpr_nodes()) > 0:
603
+ self.emit_sequence(node, u"{}")
604
+ else:
605
+ self.put(u"set()")
606
+
607
+ def visit_DictNode(self, node):
608
+ self.emit_sequence(node, u"{}")
609
+
610
+ def visit_DictItemNode(self, node):
611
+ self.visit(node.key)
612
+ self.put(u": ")
613
+ self.visit(node.value)
614
+
615
+ unop_precedence = {
616
+ 'not': 3, '!': 3,
617
+ '+': 11, '-': 11, '~': 11,
618
+ }
619
+ binop_precedence = {
620
+ 'or': 1,
621
+ 'and': 2,
622
+ # unary: 'not': 3, '!': 3,
623
+ 'in': 4, 'not_in': 4, 'is': 4, 'is_not': 4, '<': 4, '<=': 4, '>': 4, '>=': 4, '!=': 4, '==': 4,
624
+ '|': 5,
625
+ '^': 6,
626
+ '&': 7,
627
+ '<<': 8, '>>': 8,
628
+ '+': 9, '-': 9,
629
+ '*': 10, '@': 10, '/': 10, '//': 10, '%': 10,
630
+ # unary: '+': 11, '-': 11, '~': 11
631
+ '**': 12,
632
+ }
633
+
634
+ def operator_enter(self, new_prec):
635
+ old_prec = self.precedence[-1]
636
+ if old_prec > new_prec:
637
+ self.put(u"(")
638
+ self.precedence.append(new_prec)
639
+
640
+ def operator_exit(self):
641
+ old_prec, new_prec = self.precedence[-2:]
642
+ if old_prec > new_prec:
643
+ self.put(u")")
644
+ self.precedence.pop()
645
+
646
+ def visit_NotNode(self, node):
647
+ op = 'not'
648
+ prec = self.unop_precedence[op]
649
+ self.operator_enter(prec)
650
+ self.put(u"not ")
651
+ self.visit(node.operand)
652
+ self.operator_exit()
653
+
654
+ def visit_UnopNode(self, node):
655
+ op = node.operator
656
+ prec = self.unop_precedence[op]
657
+ self.operator_enter(prec)
658
+ self.put(u"%s" % node.operator)
659
+ self.visit(node.operand)
660
+ self.operator_exit()
661
+
662
+ def visit_BinopNode(self, node):
663
+ op = node.operator
664
+ prec = self.binop_precedence.get(op, 0)
665
+ self.operator_enter(prec)
666
+ self.visit(node.operand1)
667
+ self.put(u" %s " % op.replace('_', ' '))
668
+ self.visit(node.operand2)
669
+ self.operator_exit()
670
+
671
+ def visit_BoolBinopNode(self, node):
672
+ self.visit_BinopNode(node)
673
+
674
+ def visit_PrimaryCmpNode(self, node):
675
+ self.visit_BinopNode(node)
676
+
677
+ def visit_IndexNode(self, node):
678
+ self.visit(node.base)
679
+ self.put(u"[")
680
+ if isinstance(node.index, TupleNode):
681
+ self.emit_sequence(node.index)
682
+ else:
683
+ self.visit(node.index)
684
+ self.put(u"]")
685
+
686
+ def visit_SliceIndexNode(self, node):
687
+ self.visit(node.base)
688
+ self.put(u"[")
689
+ if node.start:
690
+ self.visit(node.start)
691
+ self.put(u":")
692
+ if node.stop:
693
+ self.visit(node.stop)
694
+ if node.slice:
695
+ self.put(u":")
696
+ self.visit(node.slice)
697
+ self.put(u"]")
698
+
699
+ def visit_SliceNode(self, node):
700
+ if not node.start.is_none:
701
+ self.visit(node.start)
702
+ self.put(u":")
703
+ if not node.stop.is_none:
704
+ self.visit(node.stop)
705
+ if not node.step.is_none:
706
+ self.put(u":")
707
+ self.visit(node.step)
708
+
709
+ def visit_CondExprNode(self, node):
710
+ self.visit(node.true_val)
711
+ self.put(u" if ")
712
+ self.visit(node.test)
713
+ self.put(u" else ")
714
+ self.visit(node.false_val)
715
+
716
+ def visit_AttributeNode(self, node):
717
+ self.visit(node.obj)
718
+ self.put(u".%s" % node.attribute)
719
+
720
+ def visit_SimpleCallNode(self, node):
721
+ self.visit(node.function)
722
+ self.put(u"(")
723
+ self.comma_separated_list(node.args)
724
+ self.put(")")
725
+
726
+ def emit_pos_args(self, node):
727
+ if node is None:
728
+ return
729
+ if isinstance(node, AddNode):
730
+ self.emit_pos_args(node.operand1)
731
+ self.emit_pos_args(node.operand2)
732
+ elif isinstance(node, TupleNode):
733
+ for expr in node.subexpr_nodes():
734
+ self.visit(expr)
735
+ self.put(u", ")
736
+ elif isinstance(node, AsTupleNode):
737
+ self.put("*")
738
+ self.visit(node.arg)
739
+ self.put(u", ")
740
+ else:
741
+ self.visit(node)
742
+ self.put(u", ")
743
+
744
+ def emit_kwd_args(self, node):
745
+ if node is None:
746
+ return
747
+ if isinstance(node, MergedDictNode):
748
+ for expr in node.subexpr_nodes():
749
+ self.emit_kwd_args(expr)
750
+ elif isinstance(node, DictNode):
751
+ for expr in node.subexpr_nodes():
752
+ self.put(u"%s=" % expr.key.value)
753
+ self.visit(expr.value)
754
+ self.put(u", ")
755
+ else:
756
+ self.put(u"**")
757
+ self.visit(node)
758
+ self.put(u", ")
759
+
760
+ def visit_GeneralCallNode(self, node):
761
+ self.visit(node.function)
762
+ self.put(u"(")
763
+ self.emit_pos_args(node.positional_args)
764
+ self.emit_kwd_args(node.keyword_args)
765
+ self.remove(u", ")
766
+ self.put(")")
767
+
768
+ def emit_comprehension(self, body, target,
769
+ sequence, condition,
770
+ parens=(u"", u"")):
771
+ open_paren, close_paren = parens
772
+ self.put(open_paren)
773
+ self.visit(body)
774
+ self.put(u" for ")
775
+ self.visit(target)
776
+ self.put(u" in ")
777
+ self.visit(sequence)
778
+ if condition:
779
+ self.put(u" if ")
780
+ self.visit(condition)
781
+ self.put(close_paren)
782
+
783
+ def visit_ComprehensionAppendNode(self, node):
784
+ self.visit(node.expr)
785
+
786
+ def visit_DictComprehensionAppendNode(self, node):
787
+ self.visit(node.key_expr)
788
+ self.put(u": ")
789
+ self.visit(node.value_expr)
790
+
791
+ def visit_ComprehensionNode(self, node):
792
+ tpmap = {'list': u"[]", 'dict': u"{}", 'set': u"{}"}
793
+ parens = tpmap[node.type.py_type_name()]
794
+ body = node.loop.body
795
+ target = node.loop.target
796
+ sequence = node.loop.iterator.sequence
797
+ condition = None
798
+ if hasattr(body, 'if_clauses'):
799
+ # type(body) is Nodes.IfStatNode
800
+ condition = body.if_clauses[0].condition
801
+ body = body.if_clauses[0].body
802
+ self.emit_comprehension(body, target, sequence, condition, parens)
803
+
804
+ def visit_GeneratorExpressionNode(self, node):
805
+ body = node.loop.body
806
+ target = node.loop.target
807
+ sequence = node.loop.iterator.sequence
808
+ condition = None
809
+ if hasattr(body, 'if_clauses'):
810
+ # type(body) is Nodes.IfStatNode
811
+ condition = body.if_clauses[0].condition
812
+ body = body.if_clauses[0].body.expr.arg
813
+ elif hasattr(body, 'expr'):
814
+ # type(body) is Nodes.ExprStatNode
815
+ body = body.expr.arg
816
+ self.emit_comprehension(body, target, sequence, condition, u"()")
Cython/Compiler/AnalysedTreeTransforms.py ADDED
@@ -0,0 +1,99 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ from .Visitor import ScopeTrackingTransform
4
+ from .Nodes import StatListNode, SingleAssignmentNode, CFuncDefNode, DefNode
5
+ from .ExprNodes import DictNode, DictItemNode, NameNode, UnicodeNode
6
+ from .PyrexTypes import py_object_type
7
+ from .StringEncoding import EncodedString
8
+ from . import Symtab
9
+
10
+ class AutoTestDictTransform(ScopeTrackingTransform):
11
+ # Handles autotestdict directive
12
+
13
+ blacklist = ['__cinit__', '__dealloc__', '__richcmp__',
14
+ '__nonzero__', '__bool__',
15
+ '__len__', '__contains__']
16
+
17
+ def visit_ModuleNode(self, node):
18
+ if node.is_pxd:
19
+ return node
20
+ self.scope_type = 'module'
21
+ self.scope_node = node
22
+
23
+ if not self.current_directives['autotestdict']:
24
+ return node
25
+ self.all_docstrings = self.current_directives['autotestdict.all']
26
+ self.cdef_docstrings = self.all_docstrings or self.current_directives['autotestdict.cdef']
27
+
28
+ assert isinstance(node.body, StatListNode)
29
+
30
+ # First see if __test__ is already created
31
+ if u'__test__' in node.scope.entries:
32
+ # Do nothing
33
+ return node
34
+
35
+ pos = node.pos
36
+
37
+ self.tests = []
38
+ self.testspos = node.pos
39
+
40
+ test_dict_entry = node.scope.declare_var(EncodedString(u'__test__'),
41
+ py_object_type,
42
+ pos,
43
+ visibility='public')
44
+ create_test_dict_assignment = SingleAssignmentNode(pos,
45
+ lhs=NameNode(pos, name=EncodedString(u'__test__'),
46
+ entry=test_dict_entry),
47
+ rhs=DictNode(pos, key_value_pairs=self.tests))
48
+ self.visitchildren(node)
49
+ node.body.stats.append(create_test_dict_assignment)
50
+ return node
51
+
52
+ def add_test(self, testpos, path, doctest):
53
+ pos = self.testspos
54
+ keystr = u'%s (line %d)' % (path, testpos[1])
55
+ key = UnicodeNode(pos, value=EncodedString(keystr))
56
+ value = UnicodeNode(pos, value=doctest)
57
+ self.tests.append(DictItemNode(pos, key=key, value=value))
58
+
59
+ def visit_ExprNode(self, node):
60
+ # expressions cannot contain functions and lambda expressions
61
+ # do not have a docstring
62
+ return node
63
+
64
+ def visit_FuncDefNode(self, node):
65
+ if not node.doc or (isinstance(node, DefNode) and node.fused_py_func):
66
+ return node
67
+ if not self.cdef_docstrings:
68
+ if isinstance(node, CFuncDefNode) and not node.py_func:
69
+ return node
70
+ if not self.all_docstrings and '>>>' not in node.doc:
71
+ return node
72
+
73
+ pos = self.testspos
74
+ if self.scope_type == 'module':
75
+ path = node.entry.name
76
+ elif self.scope_type in ('pyclass', 'cclass'):
77
+ if isinstance(node, CFuncDefNode):
78
+ if node.py_func is not None:
79
+ name = node.py_func.name
80
+ else:
81
+ name = node.entry.name
82
+ else:
83
+ name = node.name
84
+ if self.scope_type == 'cclass' and name in self.blacklist:
85
+ return node
86
+ if self.scope_type == 'pyclass':
87
+ class_name = self.scope_node.name
88
+ else:
89
+ class_name = self.scope_node.class_name
90
+ if isinstance(node.entry.scope, Symtab.PropertyScope):
91
+ property_method_name = node.entry.scope.name
92
+ path = "%s.%s.%s" % (class_name, node.entry.scope.name,
93
+ node.entry.name)
94
+ else:
95
+ path = "%s.%s" % (class_name, node.entry.name)
96
+ else:
97
+ assert False
98
+ self.add_test(node.pos, path, node.doc)
99
+ return node
Cython/Compiler/Annotate.py ADDED
@@ -0,0 +1,317 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Note: Work in progress
2
+
3
+ from __future__ import absolute_import
4
+
5
+ import os
6
+ import os.path
7
+ import re
8
+ import codecs
9
+ import textwrap
10
+ from datetime import datetime
11
+ from functools import partial
12
+ from collections import defaultdict
13
+ from xml.sax.saxutils import escape as html_escape
14
+ try:
15
+ from StringIO import StringIO
16
+ except ImportError:
17
+ from io import StringIO # does not support writing 'str' in Py2
18
+
19
+ from . import Version
20
+ from .Code import CCodeWriter
21
+ from .. import Utils
22
+
23
+
24
+ class AnnotationCCodeWriter(CCodeWriter):
25
+
26
+ def __init__(self, create_from=None, buffer=None, copy_formatting=True):
27
+ CCodeWriter.__init__(self, create_from, buffer, copy_formatting=copy_formatting)
28
+ if create_from is None:
29
+ self.annotation_buffer = StringIO()
30
+ self.last_annotated_pos = None
31
+ # annotations[filename][line] -> [(column, AnnotationItem)*]
32
+ self.annotations = defaultdict(partial(defaultdict, list))
33
+ # code[filename][line] -> str
34
+ self.code = defaultdict(partial(defaultdict, str))
35
+ # scopes[filename][line] -> set(scopes)
36
+ self.scopes = defaultdict(partial(defaultdict, set))
37
+ else:
38
+ # When creating an insertion point, keep references to the same database
39
+ self.annotation_buffer = create_from.annotation_buffer
40
+ self.annotations = create_from.annotations
41
+ self.code = create_from.code
42
+ self.scopes = create_from.scopes
43
+ self.last_annotated_pos = create_from.last_annotated_pos
44
+
45
+ def create_new(self, create_from, buffer, copy_formatting):
46
+ return AnnotationCCodeWriter(create_from, buffer, copy_formatting)
47
+
48
+ def write(self, s):
49
+ CCodeWriter.write(self, s)
50
+ self.annotation_buffer.write(s)
51
+
52
+ def mark_pos(self, pos, trace=True):
53
+ if pos is not None:
54
+ CCodeWriter.mark_pos(self, pos, trace)
55
+ if self.funcstate and self.funcstate.scope:
56
+ # lambdas and genexprs can result in multiple scopes per line => keep them in a set
57
+ self.scopes[pos[0].filename][pos[1]].add(self.funcstate.scope)
58
+ if self.last_annotated_pos:
59
+ source_desc, line, _ = self.last_annotated_pos
60
+ pos_code = self.code[source_desc.filename]
61
+ pos_code[line] += self.annotation_buffer.getvalue()
62
+ self.annotation_buffer = StringIO()
63
+ self.last_annotated_pos = pos
64
+
65
+ def annotate(self, pos, item):
66
+ self.annotations[pos[0].filename][pos[1]].append((pos[2], item))
67
+
68
+ def _css(self):
69
+ """css template will later allow to choose a colormap"""
70
+ css = [self._css_template]
71
+ for i in range(255):
72
+ color = u"FFFF%02x" % int(255/(1+i/10.0))
73
+ css.append('.cython.score-%d {background-color: #%s;}' % (i, color))
74
+ try:
75
+ from pygments.formatters import HtmlFormatter
76
+ except ImportError:
77
+ pass
78
+ else:
79
+ css.append(HtmlFormatter().get_style_defs('.cython'))
80
+ return '\n'.join(css)
81
+
82
+ _css_template = textwrap.dedent("""
83
+ body.cython { font-family: courier; font-size: 12; }
84
+
85
+ .cython.tag { }
86
+ .cython.line { margin: 0em }
87
+ .cython.code { font-size: 9; color: #444444; display: none; margin: 0px 0px 0px 8px; border-left: 8px none; }
88
+
89
+ .cython.line .run { background-color: #B0FFB0; }
90
+ .cython.line .mis { background-color: #FFB0B0; }
91
+ .cython.code.run { border-left: 8px solid #B0FFB0; }
92
+ .cython.code.mis { border-left: 8px solid #FFB0B0; }
93
+
94
+ .cython.code .py_c_api { color: red; }
95
+ .cython.code .py_macro_api { color: #FF7000; }
96
+ .cython.code .pyx_c_api { color: #FF3000; }
97
+ .cython.code .pyx_macro_api { color: #FF7000; }
98
+ .cython.code .refnanny { color: #FFA000; }
99
+ .cython.code .trace { color: #FFA000; }
100
+ .cython.code .error_goto { color: #FFA000; }
101
+
102
+ .cython.code .coerce { color: #008000; border: 1px dotted #008000 }
103
+ .cython.code .py_attr { color: #FF0000; font-weight: bold; }
104
+ .cython.code .c_attr { color: #0000FF; }
105
+ .cython.code .py_call { color: #FF0000; font-weight: bold; }
106
+ .cython.code .c_call { color: #0000FF; }
107
+ """)
108
+
109
+ # on-click toggle function to show/hide C source code
110
+ _onclick_attr = ' onclick="{0}"'.format((
111
+ "(function(s){"
112
+ " s.display = s.display === 'block' ? 'none' : 'block'"
113
+ "})(this.nextElementSibling.style)"
114
+ ).replace(' ', '') # poor dev's JS minification
115
+ )
116
+
117
+ def save_annotation(self, source_filename, target_filename, coverage_xml=None):
118
+ with Utils.open_source_file(source_filename) as f:
119
+ code = f.read()
120
+ generated_code = self.code.get(source_filename, {})
121
+ c_file = Utils.decode_filename(os.path.basename(target_filename))
122
+ html_filename = os.path.splitext(target_filename)[0] + ".html"
123
+
124
+ with codecs.open(html_filename, "w", encoding="UTF-8") as out_buffer:
125
+ out_buffer.write(self._save_annotation(code, generated_code, c_file, source_filename, coverage_xml))
126
+
127
+ def _save_annotation_header(self, c_file, source_filename, coverage_timestamp=None):
128
+ coverage_info = ''
129
+ if coverage_timestamp:
130
+ coverage_info = u' with coverage data from {timestamp}'.format(
131
+ timestamp=datetime.fromtimestamp(int(coverage_timestamp) // 1000))
132
+
133
+ outlist = [
134
+ textwrap.dedent(u'''\
135
+ <!DOCTYPE html>
136
+ <!-- Generated by Cython {watermark} -->
137
+ <html>
138
+ <head>
139
+ <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
140
+ <title>Cython: {filename}</title>
141
+ <style type="text/css">
142
+ {css}
143
+ </style>
144
+ </head>
145
+ <body class="cython">
146
+ <p><span style="border-bottom: solid 1px grey;">Generated by Cython {watermark}</span>{more_info}</p>
147
+ <p>
148
+ <span style="background-color: #FFFF00">Yellow lines</span> hint at Python interaction.<br />
149
+ Click on a line that starts with a "<code>+</code>" to see the C code that Cython generated for it.
150
+ </p>
151
+ ''').format(css=self._css(), watermark=Version.watermark,
152
+ filename=os.path.basename(source_filename) if source_filename else '',
153
+ more_info=coverage_info)
154
+ ]
155
+ if c_file:
156
+ outlist.append(u'<p>Raw output: <a href="%s">%s</a></p>\n' % (c_file, c_file))
157
+ return outlist
158
+
159
+ def _save_annotation_footer(self):
160
+ return (u'</body></html>\n',)
161
+
162
+ def _save_annotation(self, code, generated_code, c_file=None, source_filename=None, coverage_xml=None):
163
+ """
164
+ lines : original cython source code split by lines
165
+ generated_code : generated c code keyed by line number in original file
166
+ target filename : name of the file in which to store the generated html
167
+ c_file : filename in which the c_code has been written
168
+ """
169
+ if coverage_xml is not None and source_filename:
170
+ coverage_timestamp = coverage_xml.get('timestamp', '').strip()
171
+ covered_lines = self._get_line_coverage(coverage_xml, source_filename)
172
+ else:
173
+ coverage_timestamp = covered_lines = None
174
+ annotation_items = dict(self.annotations[source_filename])
175
+ scopes = dict(self.scopes[source_filename])
176
+
177
+ outlist = []
178
+ outlist.extend(self._save_annotation_header(c_file, source_filename, coverage_timestamp))
179
+ outlist.extend(self._save_annotation_body(code, generated_code, annotation_items, scopes, covered_lines))
180
+ outlist.extend(self._save_annotation_footer())
181
+ return ''.join(outlist)
182
+
183
+ def _get_line_coverage(self, coverage_xml, source_filename):
184
+ coverage_data = None
185
+ for entry in coverage_xml.iterfind('.//class'):
186
+ if not entry.get('filename'):
187
+ continue
188
+ if (entry.get('filename') == source_filename or
189
+ os.path.abspath(entry.get('filename')) == source_filename):
190
+ coverage_data = entry
191
+ break
192
+ elif source_filename.endswith(entry.get('filename')):
193
+ coverage_data = entry # but we might still find a better match...
194
+ if coverage_data is None:
195
+ return None
196
+ return dict(
197
+ (int(line.get('number')), int(line.get('hits')))
198
+ for line in coverage_data.iterfind('lines/line')
199
+ )
200
+
201
+ def _htmlify_code(self, code):
202
+ try:
203
+ from pygments import highlight
204
+ from pygments.lexers import CythonLexer
205
+ from pygments.formatters import HtmlFormatter
206
+ except ImportError:
207
+ # no Pygments, just escape the code
208
+ return html_escape(code)
209
+
210
+ html_code = highlight(
211
+ code, CythonLexer(stripnl=False, stripall=False),
212
+ HtmlFormatter(nowrap=True))
213
+ return html_code
214
+
215
+ def _save_annotation_body(self, cython_code, generated_code, annotation_items, scopes, covered_lines=None):
216
+ outlist = [u'<div class="cython">']
217
+ pos_comment_marker = u'/* \N{HORIZONTAL ELLIPSIS} */\n'
218
+ new_calls_map = dict(
219
+ (name, 0) for name in
220
+ 'refnanny trace py_macro_api py_c_api pyx_macro_api pyx_c_api error_goto'.split()
221
+ ).copy
222
+
223
+ self.mark_pos(None)
224
+
225
+ def annotate(match):
226
+ group_name = match.lastgroup
227
+ calls[group_name] += 1
228
+ return u"<span class='%s'>%s</span>" % (
229
+ group_name, match.group(group_name))
230
+
231
+ lines = self._htmlify_code(cython_code).splitlines()
232
+ lineno_width = len(str(len(lines)))
233
+ if not covered_lines:
234
+ covered_lines = None
235
+
236
+ for k, line in enumerate(lines, 1):
237
+ try:
238
+ c_code = generated_code[k]
239
+ except KeyError:
240
+ c_code = ''
241
+ else:
242
+ c_code = _replace_pos_comment(pos_comment_marker, c_code)
243
+ if c_code.startswith(pos_comment_marker):
244
+ c_code = c_code[len(pos_comment_marker):]
245
+ c_code = html_escape(c_code)
246
+
247
+ calls = new_calls_map()
248
+ c_code = _parse_code(annotate, c_code)
249
+ score = (5 * calls['py_c_api'] + 2 * calls['pyx_c_api'] +
250
+ calls['py_macro_api'] + calls['pyx_macro_api'])
251
+
252
+ if c_code:
253
+ onclick = self._onclick_attr
254
+ expandsymbol = '+'
255
+ else:
256
+ onclick = ''
257
+ expandsymbol = '&#xA0;'
258
+
259
+ covered = ''
260
+ if covered_lines is not None and k in covered_lines:
261
+ hits = covered_lines[k]
262
+ if hits is not None:
263
+ covered = 'run' if hits else 'mis'
264
+
265
+ outlist.append(
266
+ u'<pre class="cython line score-{score}"{onclick}>'
267
+ # generate line number with expand symbol in front,
268
+ # and the right number of digit
269
+ u'{expandsymbol}<span class="{covered}">{line:0{lineno_width}d}</span>: {code}</pre>\n'.format(
270
+ score=score,
271
+ expandsymbol=expandsymbol,
272
+ covered=covered,
273
+ lineno_width=lineno_width,
274
+ line=k,
275
+ code=line.rstrip(),
276
+ onclick=onclick,
277
+ ))
278
+ if c_code:
279
+ outlist.append(u"<pre class='cython code score-{score} {covered}'>{code}</pre>".format(
280
+ score=score, covered=covered, code=c_code))
281
+ outlist.append(u"</div>")
282
+ return outlist
283
+
284
+
285
+ _parse_code = re.compile((
286
+ br'(?P<refnanny>__Pyx_X?(?:GOT|GIVE)REF|__Pyx_RefNanny[A-Za-z]+)|'
287
+ br'(?P<trace>__Pyx_Trace[A-Za-z]+)|'
288
+ br'(?:'
289
+ br'(?P<pyx_macro_api>__Pyx_[A-Z][A-Z_]+)|'
290
+ br'(?P<pyx_c_api>(?:__Pyx_[A-Z][a-z_][A-Za-z_]*)|__pyx_convert_[A-Za-z_]*)|'
291
+ br'(?P<py_macro_api>Py[A-Z][a-z]+_[A-Z][A-Z_]+)|'
292
+ br'(?P<py_c_api>Py[A-Z][a-z]+_[A-Z][a-z][A-Za-z_]*)'
293
+ br')(?=\()|' # look-ahead to exclude subsequent '(' from replacement
294
+ br'(?P<error_goto>(?:(?<=;) *if [^;]* +)?__PYX_ERR\([^)]+\))'
295
+ ).decode('ascii')).sub
296
+
297
+
298
+ _replace_pos_comment = re.compile(
299
+ # this matches what Cython generates as code line marker comment
300
+ br'^\s*/\*(?:(?:[^*]|\*[^/])*\n)+\s*\*/\s*\n'.decode('ascii'),
301
+ re.M
302
+ ).sub
303
+
304
+
305
+ class AnnotationItem(object):
306
+
307
+ def __init__(self, style, text, tag="", size=0):
308
+ self.style = style
309
+ self.text = text
310
+ self.tag = tag
311
+ self.size = size
312
+
313
+ def start(self):
314
+ return u"<span class='cython tag %s' title='%s'>%s" % (self.style, self.text, self.tag)
315
+
316
+ def end(self):
317
+ return self.size, u"</span>"
Cython/Compiler/AutoDocTransforms.py ADDED
@@ -0,0 +1,214 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import, print_function
2
+
3
+ from .Visitor import CythonTransform
4
+ from .StringEncoding import EncodedString
5
+ from . import Options
6
+ from . import PyrexTypes, ExprNodes
7
+ from ..CodeWriter import ExpressionWriter
8
+
9
+
10
+ class AnnotationWriter(ExpressionWriter):
11
+
12
+ def visit_Node(self, node):
13
+ self.put(u"<???>")
14
+
15
+ def visit_LambdaNode(self, node):
16
+ # XXX Should we do better?
17
+ self.put("<lambda>")
18
+
19
+
20
+ class EmbedSignature(CythonTransform):
21
+
22
+ def __init__(self, context):
23
+ super(EmbedSignature, self).__init__(context)
24
+ self.class_name = None
25
+ self.class_node = None
26
+
27
+ def _fmt_expr(self, node):
28
+ writer = AnnotationWriter()
29
+ result = writer.write(node)
30
+ # print(type(node).__name__, '-->', result)
31
+ return result
32
+
33
+ def _fmt_arg(self, arg):
34
+ if arg.type is PyrexTypes.py_object_type or arg.is_self_arg:
35
+ doc = arg.name
36
+ else:
37
+ doc = arg.type.declaration_code(arg.name, for_display=1)
38
+
39
+ if arg.annotation:
40
+ annotation = self._fmt_expr(arg.annotation)
41
+ doc = doc + (': %s' % annotation)
42
+ if arg.default:
43
+ default = self._fmt_expr(arg.default)
44
+ doc = doc + (' = %s' % default)
45
+ elif arg.default:
46
+ default = self._fmt_expr(arg.default)
47
+ doc = doc + ('=%s' % default)
48
+ return doc
49
+
50
+ def _fmt_star_arg(self, arg):
51
+ arg_doc = arg.name
52
+ if arg.annotation:
53
+ annotation = self._fmt_expr(arg.annotation)
54
+ arg_doc = arg_doc + (': %s' % annotation)
55
+ return arg_doc
56
+
57
+ def _fmt_arglist(self, args,
58
+ npargs=0, pargs=None,
59
+ nkargs=0, kargs=None,
60
+ hide_self=False):
61
+ arglist = []
62
+ for arg in args:
63
+ if not hide_self or not arg.entry.is_self_arg:
64
+ arg_doc = self._fmt_arg(arg)
65
+ arglist.append(arg_doc)
66
+ if pargs:
67
+ arg_doc = self._fmt_star_arg(pargs)
68
+ arglist.insert(npargs, '*%s' % arg_doc)
69
+ elif nkargs:
70
+ arglist.insert(npargs, '*')
71
+ if kargs:
72
+ arg_doc = self._fmt_star_arg(kargs)
73
+ arglist.append('**%s' % arg_doc)
74
+ return arglist
75
+
76
+ def _fmt_ret_type(self, ret):
77
+ if ret is PyrexTypes.py_object_type:
78
+ return None
79
+ else:
80
+ return ret.declaration_code("", for_display=1)
81
+
82
+ def _fmt_signature(self, cls_name, func_name, args,
83
+ npargs=0, pargs=None,
84
+ nkargs=0, kargs=None,
85
+ return_expr=None,
86
+ return_type=None, hide_self=False):
87
+ arglist = self._fmt_arglist(args,
88
+ npargs, pargs,
89
+ nkargs, kargs,
90
+ hide_self=hide_self)
91
+ arglist_doc = ', '.join(arglist)
92
+ func_doc = '%s(%s)' % (func_name, arglist_doc)
93
+ if cls_name:
94
+ func_doc = '%s.%s' % (cls_name, func_doc)
95
+ ret_doc = None
96
+ if return_expr:
97
+ ret_doc = self._fmt_expr(return_expr)
98
+ elif return_type:
99
+ ret_doc = self._fmt_ret_type(return_type)
100
+ if ret_doc:
101
+ func_doc = '%s -> %s' % (func_doc, ret_doc)
102
+ return func_doc
103
+
104
+ def _embed_signature(self, signature, node_doc):
105
+ if node_doc:
106
+ return "%s\n%s" % (signature, node_doc)
107
+ else:
108
+ return signature
109
+
110
+ def __call__(self, node):
111
+ if not Options.docstrings:
112
+ return node
113
+ else:
114
+ return super(EmbedSignature, self).__call__(node)
115
+
116
+ def visit_ClassDefNode(self, node):
117
+ oldname = self.class_name
118
+ oldclass = self.class_node
119
+ self.class_node = node
120
+ try:
121
+ # PyClassDefNode
122
+ self.class_name = node.name
123
+ except AttributeError:
124
+ # CClassDefNode
125
+ self.class_name = node.class_name
126
+ self.visitchildren(node)
127
+ self.class_name = oldname
128
+ self.class_node = oldclass
129
+ return node
130
+
131
+ def visit_LambdaNode(self, node):
132
+ # lambda expressions so not have signature or inner functions
133
+ return node
134
+
135
+ def visit_DefNode(self, node):
136
+ if not self.current_directives['embedsignature']:
137
+ return node
138
+
139
+ is_constructor = False
140
+ hide_self = False
141
+ if node.entry.is_special:
142
+ is_constructor = self.class_node and node.name == '__init__'
143
+ if not is_constructor:
144
+ return node
145
+ class_name, func_name = None, self.class_name
146
+ hide_self = True
147
+ else:
148
+ class_name, func_name = self.class_name, node.name
149
+
150
+ nkargs = getattr(node, 'num_kwonly_args', 0)
151
+ npargs = len(node.args) - nkargs
152
+ signature = self._fmt_signature(
153
+ class_name, func_name, node.args,
154
+ npargs, node.star_arg,
155
+ nkargs, node.starstar_arg,
156
+ return_expr=node.return_type_annotation,
157
+ return_type=None, hide_self=hide_self)
158
+ if signature:
159
+ if is_constructor:
160
+ doc_holder = self.class_node.entry.type.scope
161
+ else:
162
+ doc_holder = node.entry
163
+
164
+ if doc_holder.doc is not None:
165
+ old_doc = doc_holder.doc
166
+ elif not is_constructor and getattr(node, 'py_func', None) is not None:
167
+ old_doc = node.py_func.entry.doc
168
+ else:
169
+ old_doc = None
170
+ new_doc = self._embed_signature(signature, old_doc)
171
+ doc_holder.doc = EncodedString(new_doc)
172
+ if not is_constructor and getattr(node, 'py_func', None) is not None:
173
+ node.py_func.entry.doc = EncodedString(new_doc)
174
+ return node
175
+
176
+ def visit_CFuncDefNode(self, node):
177
+ if not self.current_directives['embedsignature']:
178
+ return node
179
+ if not node.overridable: # not cpdef FOO(...):
180
+ return node
181
+
182
+ signature = self._fmt_signature(
183
+ self.class_name, node.declarator.base.name,
184
+ node.declarator.args,
185
+ return_type=node.return_type)
186
+ if signature:
187
+ if node.entry.doc is not None:
188
+ old_doc = node.entry.doc
189
+ elif getattr(node, 'py_func', None) is not None:
190
+ old_doc = node.py_func.entry.doc
191
+ else:
192
+ old_doc = None
193
+ new_doc = self._embed_signature(signature, old_doc)
194
+ node.entry.doc = EncodedString(new_doc)
195
+ if hasattr(node, 'py_func') and node.py_func is not None:
196
+ node.py_func.entry.doc = EncodedString(new_doc)
197
+ return node
198
+
199
+ def visit_PropertyNode(self, node):
200
+ if not self.current_directives['embedsignature']:
201
+ return node
202
+
203
+ entry = node.entry
204
+ if entry.visibility == 'public':
205
+ # property synthesised from a cdef public attribute
206
+ type_name = entry.type.declaration_code("", for_display=1)
207
+ if not entry.type.is_pyobject:
208
+ type_name = "'%s'" % type_name
209
+ elif entry.type.is_extension_type:
210
+ type_name = entry.type.module_name + '.' + type_name
211
+ signature = '%s: %s' % (entry.name, type_name)
212
+ new_doc = self._embed_signature(signature, entry.doc)
213
+ entry.doc = EncodedString(new_doc)
214
+ return node
Cython/Compiler/Buffer.py ADDED
@@ -0,0 +1,740 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ from .Visitor import CythonTransform
4
+ from .ModuleNode import ModuleNode
5
+ from .Errors import CompileError
6
+ from .UtilityCode import CythonUtilityCode
7
+ from .Code import UtilityCode, TempitaUtilityCode
8
+
9
+ from . import Options
10
+ from . import Interpreter
11
+ from . import PyrexTypes
12
+ from . import Naming
13
+ from . import Symtab
14
+
15
+ def dedent(text, reindent=0):
16
+ from textwrap import dedent
17
+ text = dedent(text)
18
+ if reindent > 0:
19
+ indent = " " * reindent
20
+ text = '\n'.join([indent + x for x in text.split('\n')])
21
+ return text
22
+
23
+ class IntroduceBufferAuxiliaryVars(CythonTransform):
24
+
25
+ #
26
+ # Entry point
27
+ #
28
+
29
+ buffers_exists = False
30
+ using_memoryview = False
31
+
32
+ def __call__(self, node):
33
+ assert isinstance(node, ModuleNode)
34
+ self.max_ndim = 0
35
+ result = super(IntroduceBufferAuxiliaryVars, self).__call__(node)
36
+ if self.buffers_exists:
37
+ use_bufstruct_declare_code(node.scope)
38
+ use_py2_buffer_functions(node.scope)
39
+
40
+ return result
41
+
42
+
43
+ #
44
+ # Basic operations for transforms
45
+ #
46
+ def handle_scope(self, node, scope):
47
+ # For all buffers, insert extra variables in the scope.
48
+ # The variables are also accessible from the buffer_info
49
+ # on the buffer entry
50
+ scope_items = scope.entries.items()
51
+ bufvars = [entry for name, entry in scope_items if entry.type.is_buffer]
52
+ if len(bufvars) > 0:
53
+ bufvars.sort(key=lambda entry: entry.name)
54
+ self.buffers_exists = True
55
+
56
+ memviewslicevars = [entry for name, entry in scope_items if entry.type.is_memoryviewslice]
57
+ if len(memviewslicevars) > 0:
58
+ self.buffers_exists = True
59
+
60
+
61
+ for (name, entry) in scope_items:
62
+ if name == 'memoryview' and isinstance(entry.utility_code_definition, CythonUtilityCode):
63
+ self.using_memoryview = True
64
+ break
65
+ del scope_items
66
+
67
+ if isinstance(node, ModuleNode) and len(bufvars) > 0:
68
+ # for now...note that pos is wrong
69
+ raise CompileError(node.pos, "Buffer vars not allowed in module scope")
70
+ for entry in bufvars:
71
+ if entry.type.dtype.is_ptr:
72
+ raise CompileError(node.pos, "Buffers with pointer types not yet supported.")
73
+
74
+ name = entry.name
75
+ buftype = entry.type
76
+ if buftype.ndim > Options.buffer_max_dims:
77
+ raise CompileError(node.pos,
78
+ "Buffer ndims exceeds Options.buffer_max_dims = %d" % Options.buffer_max_dims)
79
+ if buftype.ndim > self.max_ndim:
80
+ self.max_ndim = buftype.ndim
81
+
82
+ # Declare auxiliary vars
83
+ def decvar(type, prefix):
84
+ cname = scope.mangle(prefix, name)
85
+ aux_var = scope.declare_var(name=None, cname=cname,
86
+ type=type, pos=node.pos)
87
+ if entry.is_arg:
88
+ aux_var.used = True # otherwise, NameNode will mark whether it is used
89
+
90
+ return aux_var
91
+
92
+ auxvars = ((PyrexTypes.c_pyx_buffer_nd_type, Naming.pybuffernd_prefix),
93
+ (PyrexTypes.c_pyx_buffer_type, Naming.pybufferstruct_prefix))
94
+ pybuffernd, rcbuffer = [decvar(type, prefix) for (type, prefix) in auxvars]
95
+
96
+ entry.buffer_aux = Symtab.BufferAux(pybuffernd, rcbuffer)
97
+
98
+ scope.buffer_entries = bufvars
99
+ self.scope = scope
100
+
101
+ def visit_ModuleNode(self, node):
102
+ self.handle_scope(node, node.scope)
103
+ self.visitchildren(node)
104
+ return node
105
+
106
+ def visit_FuncDefNode(self, node):
107
+ self.handle_scope(node, node.local_scope)
108
+ self.visitchildren(node)
109
+ return node
110
+
111
+ #
112
+ # Analysis
113
+ #
114
+ buffer_options = ("dtype", "ndim", "mode", "negative_indices", "cast") # ordered!
115
+ buffer_defaults = {"ndim": 1, "mode": "full", "negative_indices": True, "cast": False}
116
+ buffer_positional_options_count = 1 # anything beyond this needs keyword argument
117
+
118
+ ERR_BUF_OPTION_UNKNOWN = '"%s" is not a buffer option'
119
+ ERR_BUF_TOO_MANY = 'Too many buffer options'
120
+ ERR_BUF_DUP = '"%s" buffer option already supplied'
121
+ ERR_BUF_MISSING = '"%s" missing'
122
+ ERR_BUF_MODE = 'Only allowed buffer modes are: "c", "fortran", "full", "strided" (as a compile-time string)'
123
+ ERR_BUF_NDIM = 'ndim must be a non-negative integer'
124
+ ERR_BUF_DTYPE = 'dtype must be "object", numeric type or a struct'
125
+ ERR_BUF_BOOL = '"%s" must be a boolean'
126
+
127
+ def analyse_buffer_options(globalpos, env, posargs, dictargs, defaults=None, need_complete=True):
128
+ """
129
+ Must be called during type analysis, as analyse is called
130
+ on the dtype argument.
131
+
132
+ posargs and dictargs should consist of a list and a dict
133
+ of tuples (value, pos). Defaults should be a dict of values.
134
+
135
+ Returns a dict containing all the options a buffer can have and
136
+ its value (with the positions stripped).
137
+ """
138
+ if defaults is None:
139
+ defaults = buffer_defaults
140
+
141
+ posargs, dictargs = Interpreter.interpret_compiletime_options(
142
+ posargs, dictargs, type_env=env, type_args=(0, 'dtype'))
143
+
144
+ if len(posargs) > buffer_positional_options_count:
145
+ raise CompileError(posargs[-1][1], ERR_BUF_TOO_MANY)
146
+
147
+ options = {}
148
+ for name, (value, pos) in dictargs.items():
149
+ if not name in buffer_options:
150
+ raise CompileError(pos, ERR_BUF_OPTION_UNKNOWN % name)
151
+ options[name] = value
152
+
153
+ for name, (value, pos) in zip(buffer_options, posargs):
154
+ if not name in buffer_options:
155
+ raise CompileError(pos, ERR_BUF_OPTION_UNKNOWN % name)
156
+ if name in options:
157
+ raise CompileError(pos, ERR_BUF_DUP % name)
158
+ options[name] = value
159
+
160
+ # Check that they are all there and copy defaults
161
+ for name in buffer_options:
162
+ if not name in options:
163
+ try:
164
+ options[name] = defaults[name]
165
+ except KeyError:
166
+ if need_complete:
167
+ raise CompileError(globalpos, ERR_BUF_MISSING % name)
168
+
169
+ dtype = options.get("dtype")
170
+ if dtype and dtype.is_extension_type:
171
+ raise CompileError(globalpos, ERR_BUF_DTYPE)
172
+
173
+ ndim = options.get("ndim")
174
+ if ndim and (not isinstance(ndim, int) or ndim < 0):
175
+ raise CompileError(globalpos, ERR_BUF_NDIM)
176
+
177
+ mode = options.get("mode")
178
+ if mode and not (mode in ('full', 'strided', 'c', 'fortran')):
179
+ raise CompileError(globalpos, ERR_BUF_MODE)
180
+
181
+ def assert_bool(name):
182
+ x = options.get(name)
183
+ if not isinstance(x, bool):
184
+ raise CompileError(globalpos, ERR_BUF_BOOL % name)
185
+
186
+ assert_bool('negative_indices')
187
+ assert_bool('cast')
188
+
189
+ return options
190
+
191
+
192
+ #
193
+ # Code generation
194
+ #
195
+
196
+ class BufferEntry(object):
197
+ def __init__(self, entry):
198
+ self.entry = entry
199
+ self.type = entry.type
200
+ self.cname = entry.buffer_aux.buflocal_nd_var.cname
201
+ self.buf_ptr = "%s.rcbuffer->pybuffer.buf" % self.cname
202
+ self.buf_ptr_type = entry.type.buffer_ptr_type
203
+ self.init_attributes()
204
+
205
+ def init_attributes(self):
206
+ self.shape = self.get_buf_shapevars()
207
+ self.strides = self.get_buf_stridevars()
208
+ self.suboffsets = self.get_buf_suboffsetvars()
209
+
210
+ def get_buf_suboffsetvars(self):
211
+ return self._for_all_ndim("%s.diminfo[%d].suboffsets")
212
+
213
+ def get_buf_stridevars(self):
214
+ return self._for_all_ndim("%s.diminfo[%d].strides")
215
+
216
+ def get_buf_shapevars(self):
217
+ return self._for_all_ndim("%s.diminfo[%d].shape")
218
+
219
+ def _for_all_ndim(self, s):
220
+ return [s % (self.cname, i) for i in range(self.type.ndim)]
221
+
222
+ def generate_buffer_lookup_code(self, code, index_cnames):
223
+ # Create buffer lookup and return it
224
+ # This is done via utility macros/inline functions, which vary
225
+ # according to the access mode used.
226
+ params = []
227
+ nd = self.type.ndim
228
+ mode = self.type.mode
229
+ if mode == 'full':
230
+ for i, s, o in zip(index_cnames,
231
+ self.get_buf_stridevars(),
232
+ self.get_buf_suboffsetvars()):
233
+ params.append(i)
234
+ params.append(s)
235
+ params.append(o)
236
+ funcname = "__Pyx_BufPtrFull%dd" % nd
237
+ funcgen = buf_lookup_full_code
238
+ else:
239
+ if mode == 'strided':
240
+ funcname = "__Pyx_BufPtrStrided%dd" % nd
241
+ funcgen = buf_lookup_strided_code
242
+ elif mode == 'c':
243
+ funcname = "__Pyx_BufPtrCContig%dd" % nd
244
+ funcgen = buf_lookup_c_code
245
+ elif mode == 'fortran':
246
+ funcname = "__Pyx_BufPtrFortranContig%dd" % nd
247
+ funcgen = buf_lookup_fortran_code
248
+ else:
249
+ assert False
250
+ for i, s in zip(index_cnames, self.get_buf_stridevars()):
251
+ params.append(i)
252
+ params.append(s)
253
+
254
+ # Make sure the utility code is available
255
+ if funcname not in code.globalstate.utility_codes:
256
+ code.globalstate.utility_codes.add(funcname)
257
+ protocode = code.globalstate['utility_code_proto']
258
+ defcode = code.globalstate['utility_code_def']
259
+ funcgen(protocode, defcode, name=funcname, nd=nd)
260
+
261
+ buf_ptr_type_code = self.buf_ptr_type.empty_declaration_code()
262
+ ptrcode = "%s(%s, %s, %s)" % (funcname, buf_ptr_type_code, self.buf_ptr,
263
+ ", ".join(params))
264
+ return ptrcode
265
+
266
+
267
+ def get_flags(buffer_aux, buffer_type):
268
+ flags = 'PyBUF_FORMAT'
269
+ mode = buffer_type.mode
270
+ if mode == 'full':
271
+ flags += '| PyBUF_INDIRECT'
272
+ elif mode == 'strided':
273
+ flags += '| PyBUF_STRIDES'
274
+ elif mode == 'c':
275
+ flags += '| PyBUF_C_CONTIGUOUS'
276
+ elif mode == 'fortran':
277
+ flags += '| PyBUF_F_CONTIGUOUS'
278
+ else:
279
+ assert False
280
+ if buffer_aux.writable_needed: flags += "| PyBUF_WRITABLE"
281
+ return flags
282
+
283
+ def used_buffer_aux_vars(entry):
284
+ buffer_aux = entry.buffer_aux
285
+ buffer_aux.buflocal_nd_var.used = True
286
+ buffer_aux.rcbuf_var.used = True
287
+
288
+ def put_unpack_buffer_aux_into_scope(buf_entry, code):
289
+ # Generate code to copy the needed struct info into local
290
+ # variables.
291
+ buffer_aux, mode = buf_entry.buffer_aux, buf_entry.type.mode
292
+ pybuffernd_struct = buffer_aux.buflocal_nd_var.cname
293
+
294
+ fldnames = ['strides', 'shape']
295
+ if mode == 'full':
296
+ fldnames.append('suboffsets')
297
+
298
+ ln = []
299
+ for i in range(buf_entry.type.ndim):
300
+ for fldname in fldnames:
301
+ ln.append("%s.diminfo[%d].%s = %s.rcbuffer->pybuffer.%s[%d];" % \
302
+ (pybuffernd_struct, i, fldname,
303
+ pybuffernd_struct, fldname, i))
304
+ code.putln(' '.join(ln))
305
+
306
+ def put_init_vars(entry, code):
307
+ bufaux = entry.buffer_aux
308
+ pybuffernd_struct = bufaux.buflocal_nd_var.cname
309
+ pybuffer_struct = bufaux.rcbuf_var.cname
310
+ # init pybuffer_struct
311
+ code.putln("%s.pybuffer.buf = NULL;" % pybuffer_struct)
312
+ code.putln("%s.refcount = 0;" % pybuffer_struct)
313
+ # init the buffer object
314
+ # code.put_init_var_to_py_none(entry)
315
+ # init the pybuffernd_struct
316
+ code.putln("%s.data = NULL;" % pybuffernd_struct)
317
+ code.putln("%s.rcbuffer = &%s;" % (pybuffernd_struct, pybuffer_struct))
318
+
319
+
320
+ def put_acquire_arg_buffer(entry, code, pos):
321
+ buffer_aux = entry.buffer_aux
322
+ getbuffer = get_getbuffer_call(code, entry.cname, buffer_aux, entry.type)
323
+
324
+ # Acquire any new buffer
325
+ code.putln("{")
326
+ code.putln("__Pyx_BufFmt_StackElem __pyx_stack[%d];" % entry.type.dtype.struct_nesting_depth())
327
+ code.putln(code.error_goto_if("%s == -1" % getbuffer, pos))
328
+ code.putln("}")
329
+ # An exception raised in arg parsing cannot be caught, so no
330
+ # need to care about the buffer then.
331
+ put_unpack_buffer_aux_into_scope(entry, code)
332
+
333
+
334
+ def put_release_buffer_code(code, entry):
335
+ code.globalstate.use_utility_code(acquire_utility_code)
336
+ code.putln("__Pyx_SafeReleaseBuffer(&%s.rcbuffer->pybuffer);" % entry.buffer_aux.buflocal_nd_var.cname)
337
+
338
+
339
+ def get_getbuffer_call(code, obj_cname, buffer_aux, buffer_type):
340
+ ndim = buffer_type.ndim
341
+ cast = int(buffer_type.cast)
342
+ flags = get_flags(buffer_aux, buffer_type)
343
+ pybuffernd_struct = buffer_aux.buflocal_nd_var.cname
344
+
345
+ dtype_typeinfo = get_type_information_cname(code, buffer_type.dtype)
346
+
347
+ code.globalstate.use_utility_code(acquire_utility_code)
348
+ return ("__Pyx_GetBufferAndValidate(&%(pybuffernd_struct)s.rcbuffer->pybuffer, "
349
+ "(PyObject*)%(obj_cname)s, &%(dtype_typeinfo)s, %(flags)s, %(ndim)d, "
350
+ "%(cast)d, __pyx_stack)" % locals())
351
+
352
+
353
+ def put_assign_to_buffer(lhs_cname, rhs_cname, buf_entry,
354
+ is_initialized, pos, code):
355
+ """
356
+ Generate code for reassigning a buffer variables. This only deals with getting
357
+ the buffer auxiliary structure and variables set up correctly, the assignment
358
+ itself and refcounting is the responsibility of the caller.
359
+
360
+ However, the assignment operation may throw an exception so that the reassignment
361
+ never happens.
362
+
363
+ Depending on the circumstances there are two possible outcomes:
364
+ - Old buffer released, new acquired, rhs assigned to lhs
365
+ - Old buffer released, new acquired which fails, reaqcuire old lhs buffer
366
+ (which may or may not succeed).
367
+ """
368
+
369
+ buffer_aux, buffer_type = buf_entry.buffer_aux, buf_entry.type
370
+ pybuffernd_struct = buffer_aux.buflocal_nd_var.cname
371
+ flags = get_flags(buffer_aux, buffer_type)
372
+
373
+ code.putln("{") # Set up necessary stack for getbuffer
374
+ code.putln("__Pyx_BufFmt_StackElem __pyx_stack[%d];" % buffer_type.dtype.struct_nesting_depth())
375
+
376
+ getbuffer = get_getbuffer_call(code, "%s", buffer_aux, buffer_type) # fill in object below
377
+
378
+ if is_initialized:
379
+ # Release any existing buffer
380
+ code.putln('__Pyx_SafeReleaseBuffer(&%s.rcbuffer->pybuffer);' % pybuffernd_struct)
381
+ # Acquire
382
+ retcode_cname = code.funcstate.allocate_temp(PyrexTypes.c_int_type, manage_ref=False)
383
+ code.putln("%s = %s;" % (retcode_cname, getbuffer % rhs_cname))
384
+ code.putln('if (%s) {' % (code.unlikely("%s < 0" % retcode_cname)))
385
+ # If acquisition failed, attempt to reacquire the old buffer
386
+ # before raising the exception. A failure of reacquisition
387
+ # will cause the reacquisition exception to be reported, one
388
+ # can consider working around this later.
389
+ exc_temps = tuple(code.funcstate.allocate_temp(PyrexTypes.py_object_type, manage_ref=False)
390
+ for _ in range(3))
391
+ code.putln('PyErr_Fetch(&%s, &%s, &%s);' % exc_temps)
392
+ code.putln('if (%s) {' % code.unlikely("%s == -1" % (getbuffer % lhs_cname)))
393
+ code.putln('Py_XDECREF(%s); Py_XDECREF(%s); Py_XDECREF(%s);' % exc_temps) # Do not refnanny these!
394
+ code.globalstate.use_utility_code(raise_buffer_fallback_code)
395
+ code.putln('__Pyx_RaiseBufferFallbackError();')
396
+ code.putln('} else {')
397
+ code.putln('PyErr_Restore(%s, %s, %s);' % exc_temps)
398
+ code.putln('}')
399
+ code.putln('%s = %s = %s = 0;' % exc_temps)
400
+ for t in exc_temps:
401
+ code.funcstate.release_temp(t)
402
+ code.putln('}')
403
+ # Unpack indices
404
+ put_unpack_buffer_aux_into_scope(buf_entry, code)
405
+ code.putln(code.error_goto_if_neg(retcode_cname, pos))
406
+ code.funcstate.release_temp(retcode_cname)
407
+ else:
408
+ # Our entry had no previous value, so set to None when acquisition fails.
409
+ # In this case, auxiliary vars should be set up right in initialization to a zero-buffer,
410
+ # so it suffices to set the buf field to NULL.
411
+ code.putln('if (%s) {' % code.unlikely("%s == -1" % (getbuffer % rhs_cname)))
412
+ code.putln('%s = %s; __Pyx_INCREF(Py_None); %s.rcbuffer->pybuffer.buf = NULL;' %
413
+ (lhs_cname,
414
+ PyrexTypes.typecast(buffer_type, PyrexTypes.py_object_type, "Py_None"),
415
+ pybuffernd_struct))
416
+ code.putln(code.error_goto(pos))
417
+ code.put('} else {')
418
+ # Unpack indices
419
+ put_unpack_buffer_aux_into_scope(buf_entry, code)
420
+ code.putln('}')
421
+
422
+ code.putln("}") # Release stack
423
+
424
+
425
+ def put_buffer_lookup_code(entry, index_signeds, index_cnames, directives,
426
+ pos, code, negative_indices, in_nogil_context):
427
+ """
428
+ Generates code to process indices and calculate an offset into
429
+ a buffer. Returns a C string which gives a pointer which can be
430
+ read from or written to at will (it is an expression so caller should
431
+ store it in a temporary if it is used more than once).
432
+
433
+ As the bounds checking can have any number of combinations of unsigned
434
+ arguments, smart optimizations etc. we insert it directly in the function
435
+ body. The lookup however is delegated to a inline function that is instantiated
436
+ once per ndim (lookup with suboffsets tend to get quite complicated).
437
+
438
+ entry is a BufferEntry
439
+ """
440
+ negative_indices = directives['wraparound'] and negative_indices
441
+
442
+ if directives['boundscheck']:
443
+ # Check bounds and fix negative indices.
444
+ # We allocate a temporary which is initialized to -1, meaning OK (!).
445
+ # If an error occurs, the temp is set to the index dimension the
446
+ # error is occurring at.
447
+ failed_dim_temp = code.funcstate.allocate_temp(PyrexTypes.c_int_type, manage_ref=False)
448
+ code.putln("%s = -1;" % failed_dim_temp)
449
+ for dim, (signed, cname, shape) in enumerate(zip(index_signeds, index_cnames, entry.get_buf_shapevars())):
450
+ if signed != 0:
451
+ # not unsigned, deal with negative index
452
+ code.putln("if (%s < 0) {" % cname)
453
+ if negative_indices:
454
+ code.putln("%s += %s;" % (cname, shape))
455
+ code.putln("if (%s) %s = %d;" % (
456
+ code.unlikely("%s < 0" % cname),
457
+ failed_dim_temp, dim))
458
+ else:
459
+ code.putln("%s = %d;" % (failed_dim_temp, dim))
460
+ code.put("} else ")
461
+ # check bounds in positive direction
462
+ if signed != 0:
463
+ cast = ""
464
+ else:
465
+ cast = "(size_t)"
466
+ code.putln("if (%s) %s = %d;" % (
467
+ code.unlikely("%s >= %s%s" % (cname, cast, shape)),
468
+ failed_dim_temp, dim))
469
+
470
+ if in_nogil_context:
471
+ code.globalstate.use_utility_code(raise_indexerror_nogil)
472
+ func = '__Pyx_RaiseBufferIndexErrorNogil'
473
+ else:
474
+ code.globalstate.use_utility_code(raise_indexerror_code)
475
+ func = '__Pyx_RaiseBufferIndexError'
476
+
477
+ code.putln("if (%s) {" % code.unlikely("%s != -1" % failed_dim_temp))
478
+ code.putln('%s(%s);' % (func, failed_dim_temp))
479
+ code.putln(code.error_goto(pos))
480
+ code.putln('}')
481
+ code.funcstate.release_temp(failed_dim_temp)
482
+ elif negative_indices:
483
+ # Only fix negative indices.
484
+ for signed, cname, shape in zip(index_signeds, index_cnames, entry.get_buf_shapevars()):
485
+ if signed != 0:
486
+ code.putln("if (%s < 0) %s += %s;" % (cname, cname, shape))
487
+
488
+ return entry.generate_buffer_lookup_code(code, index_cnames)
489
+
490
+
491
+ def use_bufstruct_declare_code(env):
492
+ env.use_utility_code(buffer_struct_declare_code)
493
+
494
+
495
+ def buf_lookup_full_code(proto, defin, name, nd):
496
+ """
497
+ Generates a buffer lookup function for the right number
498
+ of dimensions. The function gives back a void* at the right location.
499
+ """
500
+ # _i_ndex, _s_tride, sub_o_ffset
501
+ macroargs = ", ".join(["i%d, s%d, o%d" % (i, i, i) for i in range(nd)])
502
+ proto.putln("#define %s(type, buf, %s) (type)(%s_imp(buf, %s))" % (name, macroargs, name, macroargs))
503
+
504
+ funcargs = ", ".join(["Py_ssize_t i%d, Py_ssize_t s%d, Py_ssize_t o%d" % (i, i, i) for i in range(nd)])
505
+ proto.putln("static CYTHON_INLINE void* %s_imp(void* buf, %s);" % (name, funcargs))
506
+ defin.putln(dedent("""
507
+ static CYTHON_INLINE void* %s_imp(void* buf, %s) {
508
+ char* ptr = (char*)buf;
509
+ """) % (name, funcargs) + "".join([dedent("""\
510
+ ptr += s%d * i%d;
511
+ if (o%d >= 0) ptr = *((char**)ptr) + o%d;
512
+ """) % (i, i, i, i) for i in range(nd)]
513
+ ) + "\nreturn ptr;\n}")
514
+
515
+
516
+ def buf_lookup_strided_code(proto, defin, name, nd):
517
+ """
518
+ Generates a buffer lookup function for the right number
519
+ of dimensions. The function gives back a void* at the right location.
520
+ """
521
+ # _i_ndex, _s_tride
522
+ args = ", ".join(["i%d, s%d" % (i, i) for i in range(nd)])
523
+ offset = " + ".join(["i%d * s%d" % (i, i) for i in range(nd)])
524
+ proto.putln("#define %s(type, buf, %s) (type)((char*)buf + %s)" % (name, args, offset))
525
+
526
+
527
+ def buf_lookup_c_code(proto, defin, name, nd):
528
+ """
529
+ Similar to strided lookup, but can assume that the last dimension
530
+ doesn't need a multiplication as long as.
531
+ Still we keep the same signature for now.
532
+ """
533
+ if nd == 1:
534
+ proto.putln("#define %s(type, buf, i0, s0) ((type)buf + i0)" % name)
535
+ else:
536
+ args = ", ".join(["i%d, s%d" % (i, i) for i in range(nd)])
537
+ offset = " + ".join(["i%d * s%d" % (i, i) for i in range(nd - 1)])
538
+ proto.putln("#define %s(type, buf, %s) ((type)((char*)buf + %s) + i%d)" % (name, args, offset, nd - 1))
539
+
540
+
541
+ def buf_lookup_fortran_code(proto, defin, name, nd):
542
+ """
543
+ Like C lookup, but the first index is optimized instead.
544
+ """
545
+ if nd == 1:
546
+ proto.putln("#define %s(type, buf, i0, s0) ((type)buf + i0)" % name)
547
+ else:
548
+ args = ", ".join(["i%d, s%d" % (i, i) for i in range(nd)])
549
+ offset = " + ".join(["i%d * s%d" % (i, i) for i in range(1, nd)])
550
+ proto.putln("#define %s(type, buf, %s) ((type)((char*)buf + %s) + i%d)" % (name, args, offset, 0))
551
+
552
+
553
+ def use_py2_buffer_functions(env):
554
+ env.use_utility_code(GetAndReleaseBufferUtilityCode())
555
+
556
+
557
+ class GetAndReleaseBufferUtilityCode(object):
558
+ # Emulation of PyObject_GetBuffer and PyBuffer_Release for Python 2.
559
+ # For >= 2.6 we do double mode -- use the new buffer interface on objects
560
+ # which has the right tp_flags set, but emulation otherwise.
561
+
562
+ requires = None
563
+ is_cython_utility = False
564
+
565
+ def __init__(self):
566
+ pass
567
+
568
+ def __eq__(self, other):
569
+ return isinstance(other, GetAndReleaseBufferUtilityCode)
570
+
571
+ def __hash__(self):
572
+ return 24342342
573
+
574
+ def get_tree(self, **kwargs): pass
575
+
576
+ def put_code(self, output):
577
+ code = output['utility_code_def']
578
+ proto_code = output['utility_code_proto']
579
+ env = output.module_node.scope
580
+ cython_scope = env.context.cython_scope
581
+
582
+ # Search all types for __getbuffer__ overloads
583
+ types = []
584
+ visited_scopes = set()
585
+ def find_buffer_types(scope):
586
+ if scope in visited_scopes:
587
+ return
588
+ visited_scopes.add(scope)
589
+ for m in scope.cimported_modules:
590
+ find_buffer_types(m)
591
+ for e in scope.type_entries:
592
+ if isinstance(e.utility_code_definition, CythonUtilityCode):
593
+ continue
594
+ t = e.type
595
+ if t.is_extension_type:
596
+ if scope is cython_scope and not e.used:
597
+ continue
598
+ release = get = None
599
+ for x in t.scope.pyfunc_entries:
600
+ if x.name == u"__getbuffer__": get = x.func_cname
601
+ elif x.name == u"__releasebuffer__": release = x.func_cname
602
+ if get:
603
+ types.append((t.typeptr_cname, get, release))
604
+
605
+ find_buffer_types(env)
606
+
607
+ util_code = TempitaUtilityCode.load(
608
+ "GetAndReleaseBuffer", from_file="Buffer.c",
609
+ context=dict(types=types))
610
+
611
+ proto = util_code.format_code(util_code.proto)
612
+ impl = util_code.format_code(
613
+ util_code.inject_string_constants(util_code.impl, output)[1])
614
+
615
+ proto_code.putln(proto)
616
+ code.putln(impl)
617
+
618
+
619
+ def mangle_dtype_name(dtype):
620
+ # Use prefixes to separate user defined types from builtins
621
+ # (consider "typedef float unsigned_int")
622
+ if dtype.is_pyobject:
623
+ return "object"
624
+ elif dtype.is_ptr:
625
+ return "ptr"
626
+ else:
627
+ if dtype.is_typedef or dtype.is_struct_or_union:
628
+ prefix = "nn_"
629
+ else:
630
+ prefix = ""
631
+ return prefix + dtype.specialization_name()
632
+
633
+ def get_type_information_cname(code, dtype, maxdepth=None):
634
+ """
635
+ Output the run-time type information (__Pyx_TypeInfo) for given dtype,
636
+ and return the name of the type info struct.
637
+
638
+ Structs with two floats of the same size are encoded as complex numbers.
639
+ One can separate between complex numbers declared as struct or with native
640
+ encoding by inspecting to see if the fields field of the type is
641
+ filled in.
642
+ """
643
+ namesuffix = mangle_dtype_name(dtype)
644
+ name = "__Pyx_TypeInfo_%s" % namesuffix
645
+ structinfo_name = "__Pyx_StructFields_%s" % namesuffix
646
+
647
+ if dtype.is_error: return "<error>"
648
+
649
+ # It's critical that walking the type info doesn't use more stack
650
+ # depth than dtype.struct_nesting_depth() returns, so use an assertion for this
651
+ if maxdepth is None: maxdepth = dtype.struct_nesting_depth()
652
+ if maxdepth <= 0:
653
+ assert False
654
+
655
+ if name not in code.globalstate.utility_codes:
656
+ code.globalstate.utility_codes.add(name)
657
+ typecode = code.globalstate['typeinfo']
658
+
659
+ arraysizes = []
660
+ if dtype.is_array:
661
+ while dtype.is_array:
662
+ arraysizes.append(dtype.size)
663
+ dtype = dtype.base_type
664
+
665
+ complex_possible = dtype.is_struct_or_union and dtype.can_be_complex()
666
+
667
+ declcode = dtype.empty_declaration_code()
668
+ if dtype.is_simple_buffer_dtype():
669
+ structinfo_name = "NULL"
670
+ elif dtype.is_struct:
671
+ struct_scope = dtype.scope
672
+ if dtype.is_const:
673
+ struct_scope = struct_scope.const_base_type_scope
674
+ # Must pre-call all used types in order not to recurse during utility code writing.
675
+ fields = struct_scope.var_entries
676
+ assert len(fields) > 0
677
+ types = [get_type_information_cname(code, f.type, maxdepth - 1)
678
+ for f in fields]
679
+ typecode.putln("static __Pyx_StructField %s[] = {" % structinfo_name, safe=True)
680
+ for f, typeinfo in zip(fields, types):
681
+ typecode.putln(' {&%s, "%s", offsetof(%s, %s)},' %
682
+ (typeinfo, f.name, dtype.empty_declaration_code(), f.cname), safe=True)
683
+ typecode.putln(' {NULL, NULL, 0}', safe=True)
684
+ typecode.putln("};", safe=True)
685
+ else:
686
+ assert False
687
+
688
+ rep = str(dtype)
689
+
690
+ flags = "0"
691
+ is_unsigned = "0"
692
+ if dtype is PyrexTypes.c_char_type:
693
+ is_unsigned = "IS_UNSIGNED(%s)" % declcode
694
+ typegroup = "'H'"
695
+ elif dtype.is_int:
696
+ is_unsigned = "IS_UNSIGNED(%s)" % declcode
697
+ typegroup = "%s ? 'U' : 'I'" % is_unsigned
698
+ elif complex_possible or dtype.is_complex:
699
+ typegroup = "'C'"
700
+ elif dtype.is_float:
701
+ typegroup = "'R'"
702
+ elif dtype.is_struct:
703
+ typegroup = "'S'"
704
+ if dtype.packed:
705
+ flags = "__PYX_BUF_FLAGS_PACKED_STRUCT"
706
+ elif dtype.is_pyobject:
707
+ typegroup = "'O'"
708
+ else:
709
+ assert False, dtype
710
+
711
+ typeinfo = ('static __Pyx_TypeInfo %s = '
712
+ '{ "%s", %s, sizeof(%s), { %s }, %s, %s, %s, %s };')
713
+ tup = (name, rep, structinfo_name, declcode,
714
+ ', '.join([str(x) for x in arraysizes]) or '0', len(arraysizes),
715
+ typegroup, is_unsigned, flags)
716
+ typecode.putln(typeinfo % tup, safe=True)
717
+
718
+ return name
719
+
720
+ def load_buffer_utility(util_code_name, context=None, **kwargs):
721
+ if context is None:
722
+ return UtilityCode.load(util_code_name, "Buffer.c", **kwargs)
723
+ else:
724
+ return TempitaUtilityCode.load(util_code_name, "Buffer.c", context=context, **kwargs)
725
+
726
+ context = dict(max_dims=Options.buffer_max_dims)
727
+ buffer_struct_declare_code = load_buffer_utility("BufferStructDeclare", context=context)
728
+ buffer_formats_declare_code = load_buffer_utility("BufferFormatStructs")
729
+
730
+ # Utility function to set the right exception
731
+ # The caller should immediately goto_error
732
+ raise_indexerror_code = load_buffer_utility("BufferIndexError")
733
+ raise_indexerror_nogil = load_buffer_utility("BufferIndexErrorNogil")
734
+ raise_buffer_fallback_code = load_buffer_utility("BufferFallbackError")
735
+
736
+ acquire_utility_code = load_buffer_utility("BufferGetAndValidate", context=context)
737
+ buffer_format_check_code = load_buffer_utility("BufferFormatCheck", context=context)
738
+
739
+ # See utility code BufferFormatFromTypeInfo
740
+ _typeinfo_to_format_code = load_buffer_utility("TypeInfoToFormat")
Cython/Compiler/Builtin.py ADDED
@@ -0,0 +1,449 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #
2
+ # Builtin Definitions
3
+ #
4
+
5
+ from __future__ import absolute_import
6
+
7
+ from .Symtab import BuiltinScope, StructOrUnionScope
8
+ from .Code import UtilityCode
9
+ from .TypeSlots import Signature
10
+ from . import PyrexTypes
11
+ from . import Options
12
+
13
+
14
+ # C-level implementations of builtin types, functions and methods
15
+
16
+ iter_next_utility_code = UtilityCode.load("IterNext", "ObjectHandling.c")
17
+ getattr_utility_code = UtilityCode.load("GetAttr", "ObjectHandling.c")
18
+ getattr3_utility_code = UtilityCode.load("GetAttr3", "Builtins.c")
19
+ pyexec_utility_code = UtilityCode.load("PyExec", "Builtins.c")
20
+ pyexec_globals_utility_code = UtilityCode.load("PyExecGlobals", "Builtins.c")
21
+ globals_utility_code = UtilityCode.load("Globals", "Builtins.c")
22
+
23
+ builtin_utility_code = {
24
+ 'StopAsyncIteration': UtilityCode.load_cached("StopAsyncIteration", "Coroutine.c"),
25
+ }
26
+
27
+
28
+ # mapping from builtins to their C-level equivalents
29
+
30
+ class _BuiltinOverride(object):
31
+ def __init__(self, py_name, args, ret_type, cname, py_equiv="*",
32
+ utility_code=None, sig=None, func_type=None,
33
+ is_strict_signature=False, builtin_return_type=None):
34
+ self.py_name, self.cname, self.py_equiv = py_name, cname, py_equiv
35
+ self.args, self.ret_type = args, ret_type
36
+ self.func_type, self.sig = func_type, sig
37
+ self.builtin_return_type = builtin_return_type
38
+ self.is_strict_signature = is_strict_signature
39
+ self.utility_code = utility_code
40
+
41
+ def build_func_type(self, sig=None, self_arg=None):
42
+ if sig is None:
43
+ sig = Signature(self.args, self.ret_type)
44
+ sig.exception_check = False # not needed for the current builtins
45
+ func_type = sig.function_type(self_arg)
46
+ if self.is_strict_signature:
47
+ func_type.is_strict_signature = True
48
+ if self.builtin_return_type:
49
+ func_type.return_type = builtin_types[self.builtin_return_type]
50
+ return func_type
51
+
52
+
53
+ class BuiltinAttribute(object):
54
+ def __init__(self, py_name, cname=None, field_type=None, field_type_name=None):
55
+ self.py_name = py_name
56
+ self.cname = cname or py_name
57
+ self.field_type_name = field_type_name # can't do the lookup before the type is declared!
58
+ self.field_type = field_type
59
+
60
+ def declare_in_type(self, self_type):
61
+ if self.field_type_name is not None:
62
+ # lazy type lookup
63
+ field_type = builtin_scope.lookup(self.field_type_name).type
64
+ else:
65
+ field_type = self.field_type or PyrexTypes.py_object_type
66
+ entry = self_type.scope.declare(self.py_name, self.cname, field_type, None, 'private')
67
+ entry.is_variable = True
68
+
69
+
70
+ class BuiltinFunction(_BuiltinOverride):
71
+ def declare_in_scope(self, scope):
72
+ func_type, sig = self.func_type, self.sig
73
+ if func_type is None:
74
+ func_type = self.build_func_type(sig)
75
+ scope.declare_builtin_cfunction(self.py_name, func_type, self.cname,
76
+ self.py_equiv, self.utility_code)
77
+
78
+
79
+ class BuiltinMethod(_BuiltinOverride):
80
+ def declare_in_type(self, self_type):
81
+ method_type, sig = self.func_type, self.sig
82
+ if method_type is None:
83
+ # override 'self' type (first argument)
84
+ self_arg = PyrexTypes.CFuncTypeArg("", self_type, None)
85
+ self_arg.not_none = True
86
+ self_arg.accept_builtin_subtypes = True
87
+ method_type = self.build_func_type(sig, self_arg)
88
+ self_type.scope.declare_builtin_cfunction(
89
+ self.py_name, method_type, self.cname, utility_code=self.utility_code)
90
+
91
+
92
+ builtin_function_table = [
93
+ # name, args, return, C API func, py equiv = "*"
94
+ BuiltinFunction('abs', "d", "d", "fabs",
95
+ is_strict_signature = True),
96
+ BuiltinFunction('abs', "f", "f", "fabsf",
97
+ is_strict_signature = True),
98
+ BuiltinFunction('abs', "i", "i", "abs",
99
+ is_strict_signature = True),
100
+ BuiltinFunction('abs', "l", "l", "labs",
101
+ is_strict_signature = True),
102
+ BuiltinFunction('abs', None, None, "__Pyx_abs_longlong",
103
+ utility_code = UtilityCode.load("abs_longlong", "Builtins.c"),
104
+ func_type = PyrexTypes.CFuncType(
105
+ PyrexTypes.c_longlong_type, [
106
+ PyrexTypes.CFuncTypeArg("arg", PyrexTypes.c_longlong_type, None)
107
+ ],
108
+ is_strict_signature = True, nogil=True)),
109
+ ] + list(
110
+ BuiltinFunction('abs', None, None, "/*abs_{0}*/".format(t.specialization_name()),
111
+ func_type = PyrexTypes.CFuncType(
112
+ t,
113
+ [PyrexTypes.CFuncTypeArg("arg", t, None)],
114
+ is_strict_signature = True, nogil=True))
115
+ for t in (PyrexTypes.c_uint_type, PyrexTypes.c_ulong_type, PyrexTypes.c_ulonglong_type)
116
+ ) + list(
117
+ BuiltinFunction('abs', None, None, "__Pyx_c_abs{0}".format(t.funcsuffix),
118
+ func_type = PyrexTypes.CFuncType(
119
+ t.real_type, [
120
+ PyrexTypes.CFuncTypeArg("arg", t, None)
121
+ ],
122
+ is_strict_signature = True, nogil=True))
123
+ for t in (PyrexTypes.c_float_complex_type,
124
+ PyrexTypes.c_double_complex_type,
125
+ PyrexTypes.c_longdouble_complex_type)
126
+ ) + [
127
+ BuiltinFunction('abs', "O", "O", "__Pyx_PyNumber_Absolute",
128
+ utility_code=UtilityCode.load("py_abs", "Builtins.c")),
129
+ #('all', "", "", ""),
130
+ #('any', "", "", ""),
131
+ #('ascii', "", "", ""),
132
+ #('bin', "", "", ""),
133
+ BuiltinFunction('callable', "O", "b", "__Pyx_PyCallable_Check",
134
+ utility_code = UtilityCode.load("CallableCheck", "ObjectHandling.c")),
135
+ #('chr', "", "", ""),
136
+ #('cmp', "", "", "", ""), # int PyObject_Cmp(PyObject *o1, PyObject *o2, int *result)
137
+ #('compile', "", "", ""), # PyObject* Py_CompileString( char *str, char *filename, int start)
138
+ BuiltinFunction('delattr', "OO", "r", "PyObject_DelAttr"),
139
+ BuiltinFunction('dir', "O", "O", "PyObject_Dir"),
140
+ BuiltinFunction('divmod', "OO", "O", "PyNumber_Divmod"),
141
+ BuiltinFunction('exec', "O", "O", "__Pyx_PyExecGlobals",
142
+ utility_code = pyexec_globals_utility_code),
143
+ BuiltinFunction('exec', "OO", "O", "__Pyx_PyExec2",
144
+ utility_code = pyexec_utility_code),
145
+ BuiltinFunction('exec', "OOO", "O", "__Pyx_PyExec3",
146
+ utility_code = pyexec_utility_code),
147
+ #('eval', "", "", ""),
148
+ #('execfile', "", "", ""),
149
+ #('filter', "", "", ""),
150
+ BuiltinFunction('getattr3', "OOO", "O", "__Pyx_GetAttr3", "getattr",
151
+ utility_code=getattr3_utility_code), # Pyrex legacy
152
+ BuiltinFunction('getattr', "OOO", "O", "__Pyx_GetAttr3",
153
+ utility_code=getattr3_utility_code),
154
+ BuiltinFunction('getattr', "OO", "O", "__Pyx_GetAttr",
155
+ utility_code=getattr_utility_code),
156
+ BuiltinFunction('hasattr', "OO", "b", "__Pyx_HasAttr",
157
+ utility_code = UtilityCode.load("HasAttr", "Builtins.c")),
158
+ BuiltinFunction('hash', "O", "h", "PyObject_Hash"),
159
+ #('hex', "", "", ""),
160
+ #('id', "", "", ""),
161
+ #('input', "", "", ""),
162
+ BuiltinFunction('intern', "O", "O", "__Pyx_Intern",
163
+ utility_code = UtilityCode.load("Intern", "Builtins.c")),
164
+ BuiltinFunction('isinstance', "OO", "b", "PyObject_IsInstance"),
165
+ BuiltinFunction('issubclass', "OO", "b", "PyObject_IsSubclass"),
166
+ BuiltinFunction('iter', "OO", "O", "PyCallIter_New"),
167
+ BuiltinFunction('iter', "O", "O", "PyObject_GetIter"),
168
+ BuiltinFunction('len', "O", "z", "PyObject_Length"),
169
+ BuiltinFunction('locals', "", "O", "__pyx_locals"),
170
+ #('map', "", "", ""),
171
+ #('max', "", "", ""),
172
+ #('min', "", "", ""),
173
+ BuiltinFunction('next', "O", "O", "__Pyx_PyIter_Next",
174
+ utility_code = iter_next_utility_code), # not available in Py2 => implemented here
175
+ BuiltinFunction('next', "OO", "O", "__Pyx_PyIter_Next2",
176
+ utility_code = iter_next_utility_code), # not available in Py2 => implemented here
177
+ #('oct', "", "", ""),
178
+ #('open', "ss", "O", "PyFile_FromString"), # not in Py3
179
+ ] + [
180
+ BuiltinFunction('ord', None, None, "__Pyx_long_cast",
181
+ func_type=PyrexTypes.CFuncType(
182
+ PyrexTypes.c_long_type, [PyrexTypes.CFuncTypeArg("c", c_type, None)],
183
+ is_strict_signature=True))
184
+ for c_type in [PyrexTypes.c_py_ucs4_type, PyrexTypes.c_py_unicode_type]
185
+ ] + [
186
+ BuiltinFunction('ord', None, None, "__Pyx_uchar_cast",
187
+ func_type=PyrexTypes.CFuncType(
188
+ PyrexTypes.c_uchar_type, [PyrexTypes.CFuncTypeArg("c", c_type, None)],
189
+ is_strict_signature=True))
190
+ for c_type in [PyrexTypes.c_char_type, PyrexTypes.c_schar_type, PyrexTypes.c_uchar_type]
191
+ ] + [
192
+ BuiltinFunction('ord', None, None, "__Pyx_PyObject_Ord",
193
+ utility_code=UtilityCode.load_cached("object_ord", "Builtins.c"),
194
+ func_type=PyrexTypes.CFuncType(
195
+ PyrexTypes.c_long_type, [
196
+ PyrexTypes.CFuncTypeArg("c", PyrexTypes.py_object_type, None)
197
+ ],
198
+ exception_value="(long)(Py_UCS4)-1")),
199
+ BuiltinFunction('pow', "OOO", "O", "PyNumber_Power"),
200
+ BuiltinFunction('pow', "OO", "O", "__Pyx_PyNumber_Power2",
201
+ utility_code = UtilityCode.load("pow2", "Builtins.c")),
202
+ #('range', "", "", ""),
203
+ #('raw_input', "", "", ""),
204
+ #('reduce', "", "", ""),
205
+ BuiltinFunction('reload', "O", "O", "PyImport_ReloadModule"),
206
+ BuiltinFunction('repr', "O", "O", "PyObject_Repr"), # , builtin_return_type='str'), # add in Cython 3.1
207
+ #('round', "", "", ""),
208
+ BuiltinFunction('setattr', "OOO", "r", "PyObject_SetAttr"),
209
+ #('sum', "", "", ""),
210
+ #('sorted', "", "", ""),
211
+ #('type', "O", "O", "PyObject_Type"),
212
+ #('unichr', "", "", ""),
213
+ #('unicode', "", "", ""),
214
+ #('vars', "", "", ""),
215
+ #('zip', "", "", ""),
216
+ # Can't do these easily until we have builtin type entries.
217
+ #('typecheck', "OO", "i", "PyObject_TypeCheck", False),
218
+ #('issubtype', "OO", "i", "PyType_IsSubtype", False),
219
+
220
+ # Put in namespace append optimization.
221
+ BuiltinFunction('__Pyx_PyObject_Append', "OO", "O", "__Pyx_PyObject_Append"),
222
+
223
+ # This is conditionally looked up based on a compiler directive.
224
+ BuiltinFunction('__Pyx_Globals', "", "O", "__Pyx_Globals",
225
+ utility_code=globals_utility_code),
226
+ ]
227
+
228
+
229
+ # Builtin types
230
+ # bool
231
+ # buffer
232
+ # classmethod
233
+ # dict
234
+ # enumerate
235
+ # file
236
+ # float
237
+ # int
238
+ # list
239
+ # long
240
+ # object
241
+ # property
242
+ # slice
243
+ # staticmethod
244
+ # super
245
+ # str
246
+ # tuple
247
+ # type
248
+ # xrange
249
+
250
+ builtin_types_table = [
251
+
252
+ ("type", "PyType_Type", []),
253
+
254
+ # This conflicts with the C++ bool type, and unfortunately
255
+ # C++ is too liberal about PyObject* <-> bool conversions,
256
+ # resulting in unintuitive runtime behavior and segfaults.
257
+ # ("bool", "PyBool_Type", []),
258
+
259
+ ("int", "PyInt_Type", []),
260
+ ("long", "PyLong_Type", []),
261
+ ("float", "PyFloat_Type", []),
262
+
263
+ ("complex", "PyComplex_Type", [BuiltinAttribute('cval', field_type_name = 'Py_complex'),
264
+ BuiltinAttribute('real', 'cval.real', field_type = PyrexTypes.c_double_type),
265
+ BuiltinAttribute('imag', 'cval.imag', field_type = PyrexTypes.c_double_type),
266
+ ]),
267
+
268
+ ("basestring", "PyBaseString_Type", [
269
+ BuiltinMethod("join", "TO", "T", "__Pyx_PyBaseString_Join",
270
+ utility_code=UtilityCode.load("StringJoin", "StringTools.c")),
271
+ ]),
272
+ ("bytearray", "PyByteArray_Type", [
273
+ ]),
274
+ ("bytes", "PyBytes_Type", [BuiltinMethod("__contains__", "TO", "b", "PySequence_Contains"),
275
+ BuiltinMethod("join", "TO", "O", "__Pyx_PyBytes_Join",
276
+ utility_code=UtilityCode.load("StringJoin", "StringTools.c")),
277
+ ]),
278
+ ("str", "PyString_Type", [BuiltinMethod("__contains__", "TO", "b", "PySequence_Contains"),
279
+ BuiltinMethod("join", "TO", "O", "__Pyx_PyString_Join",
280
+ builtin_return_type='basestring',
281
+ utility_code=UtilityCode.load("StringJoin", "StringTools.c")),
282
+ ]),
283
+ ("unicode", "PyUnicode_Type", [BuiltinMethod("__contains__", "TO", "b", "PyUnicode_Contains"),
284
+ BuiltinMethod("join", "TO", "T", "PyUnicode_Join"),
285
+ ]),
286
+
287
+ ("tuple", "PyTuple_Type", [BuiltinMethod("__contains__", "TO", "b", "PySequence_Contains"),
288
+ ]),
289
+
290
+ ("list", "PyList_Type", [BuiltinMethod("__contains__", "TO", "b", "PySequence_Contains"),
291
+ BuiltinMethod("insert", "TzO", "r", "PyList_Insert"),
292
+ BuiltinMethod("reverse", "T", "r", "PyList_Reverse"),
293
+ BuiltinMethod("append", "TO", "r", "__Pyx_PyList_Append",
294
+ utility_code=UtilityCode.load("ListAppend", "Optimize.c")),
295
+ BuiltinMethod("extend", "TO", "r", "__Pyx_PyList_Extend",
296
+ utility_code=UtilityCode.load("ListExtend", "Optimize.c")),
297
+ ]),
298
+
299
+ ("dict", "PyDict_Type", [BuiltinMethod("__contains__", "TO", "b", "PyDict_Contains"),
300
+ BuiltinMethod("has_key", "TO", "b", "PyDict_Contains"),
301
+ BuiltinMethod("items", "T", "O", "__Pyx_PyDict_Items",
302
+ utility_code=UtilityCode.load("py_dict_items", "Builtins.c")),
303
+ BuiltinMethod("keys", "T", "O", "__Pyx_PyDict_Keys",
304
+ utility_code=UtilityCode.load("py_dict_keys", "Builtins.c")),
305
+ BuiltinMethod("values", "T", "O", "__Pyx_PyDict_Values",
306
+ utility_code=UtilityCode.load("py_dict_values", "Builtins.c")),
307
+ BuiltinMethod("iteritems", "T", "O", "__Pyx_PyDict_IterItems",
308
+ utility_code=UtilityCode.load("py_dict_iteritems", "Builtins.c")),
309
+ BuiltinMethod("iterkeys", "T", "O", "__Pyx_PyDict_IterKeys",
310
+ utility_code=UtilityCode.load("py_dict_iterkeys", "Builtins.c")),
311
+ BuiltinMethod("itervalues", "T", "O", "__Pyx_PyDict_IterValues",
312
+ utility_code=UtilityCode.load("py_dict_itervalues", "Builtins.c")),
313
+ BuiltinMethod("viewitems", "T", "O", "__Pyx_PyDict_ViewItems",
314
+ utility_code=UtilityCode.load("py_dict_viewitems", "Builtins.c")),
315
+ BuiltinMethod("viewkeys", "T", "O", "__Pyx_PyDict_ViewKeys",
316
+ utility_code=UtilityCode.load("py_dict_viewkeys", "Builtins.c")),
317
+ BuiltinMethod("viewvalues", "T", "O", "__Pyx_PyDict_ViewValues",
318
+ utility_code=UtilityCode.load("py_dict_viewvalues", "Builtins.c")),
319
+ BuiltinMethod("clear", "T", "r", "__Pyx_PyDict_Clear",
320
+ utility_code=UtilityCode.load("py_dict_clear", "Optimize.c")),
321
+ BuiltinMethod("copy", "T", "T", "PyDict_Copy")]),
322
+
323
+ ("slice", "PySlice_Type", [BuiltinAttribute('start'),
324
+ BuiltinAttribute('stop'),
325
+ BuiltinAttribute('step'),
326
+ ]),
327
+ # ("file", "PyFile_Type", []), # not in Py3
328
+
329
+ ("set", "PySet_Type", [BuiltinMethod("__contains__", "TO", "b", "PySequence_Contains"),
330
+ BuiltinMethod("clear", "T", "r", "PySet_Clear"),
331
+ # discard() and remove() have a special treatment for unhashable values
332
+ BuiltinMethod("discard", "TO", "r", "__Pyx_PySet_Discard",
333
+ utility_code=UtilityCode.load("py_set_discard", "Optimize.c")),
334
+ BuiltinMethod("remove", "TO", "r", "__Pyx_PySet_Remove",
335
+ utility_code=UtilityCode.load("py_set_remove", "Optimize.c")),
336
+ # update is actually variadic (see Github issue #1645)
337
+ # BuiltinMethod("update", "TO", "r", "__Pyx_PySet_Update",
338
+ # utility_code=UtilityCode.load_cached("PySet_Update", "Builtins.c")),
339
+ BuiltinMethod("add", "TO", "r", "PySet_Add"),
340
+ BuiltinMethod("pop", "T", "O", "PySet_Pop")]),
341
+ ("frozenset", "PyFrozenSet_Type", []),
342
+ ("Exception", "((PyTypeObject*)PyExc_Exception)[0]", []),
343
+ ("StopAsyncIteration", "((PyTypeObject*)__Pyx_PyExc_StopAsyncIteration)[0]", []),
344
+ ]
345
+
346
+
347
+ types_that_construct_their_instance = set([
348
+ # some builtin types do not always return an instance of
349
+ # themselves - these do:
350
+ 'type', 'bool', 'long', 'float', 'complex',
351
+ 'bytes', 'unicode', 'bytearray',
352
+ 'tuple', 'list', 'dict', 'set', 'frozenset'
353
+ # 'str', # only in Py3.x
354
+ # 'file', # only in Py2.x
355
+ ])
356
+
357
+
358
+ builtin_structs_table = [
359
+ ('Py_buffer', 'Py_buffer',
360
+ [("buf", PyrexTypes.c_void_ptr_type),
361
+ ("obj", PyrexTypes.py_object_type),
362
+ ("len", PyrexTypes.c_py_ssize_t_type),
363
+ ("itemsize", PyrexTypes.c_py_ssize_t_type),
364
+ ("readonly", PyrexTypes.c_bint_type),
365
+ ("ndim", PyrexTypes.c_int_type),
366
+ ("format", PyrexTypes.c_char_ptr_type),
367
+ ("shape", PyrexTypes.c_py_ssize_t_ptr_type),
368
+ ("strides", PyrexTypes.c_py_ssize_t_ptr_type),
369
+ ("suboffsets", PyrexTypes.c_py_ssize_t_ptr_type),
370
+ ("smalltable", PyrexTypes.CArrayType(PyrexTypes.c_py_ssize_t_type, 2)),
371
+ ("internal", PyrexTypes.c_void_ptr_type),
372
+ ]),
373
+ ('Py_complex', 'Py_complex',
374
+ [('real', PyrexTypes.c_double_type),
375
+ ('imag', PyrexTypes.c_double_type),
376
+ ])
377
+ ]
378
+
379
+ # set up builtin scope
380
+
381
+ builtin_scope = BuiltinScope()
382
+
383
+ def init_builtin_funcs():
384
+ for bf in builtin_function_table:
385
+ bf.declare_in_scope(builtin_scope)
386
+
387
+ builtin_types = {}
388
+
389
+ def init_builtin_types():
390
+ global builtin_types
391
+ for name, cname, methods in builtin_types_table:
392
+ utility = builtin_utility_code.get(name)
393
+ if name == 'frozenset':
394
+ objstruct_cname = 'PySetObject'
395
+ elif name == 'bytearray':
396
+ objstruct_cname = 'PyByteArrayObject'
397
+ elif name == 'bool':
398
+ objstruct_cname = None
399
+ elif name == 'Exception':
400
+ objstruct_cname = "PyBaseExceptionObject"
401
+ elif name == 'StopAsyncIteration':
402
+ objstruct_cname = "PyBaseExceptionObject"
403
+ else:
404
+ objstruct_cname = 'Py%sObject' % name.capitalize()
405
+ the_type = builtin_scope.declare_builtin_type(name, cname, utility, objstruct_cname)
406
+ builtin_types[name] = the_type
407
+ for method in methods:
408
+ method.declare_in_type(the_type)
409
+
410
+ def init_builtin_structs():
411
+ for name, cname, attribute_types in builtin_structs_table:
412
+ scope = StructOrUnionScope(name)
413
+ for attribute_name, attribute_type in attribute_types:
414
+ scope.declare_var(attribute_name, attribute_type, None,
415
+ attribute_name, allow_pyobject=True)
416
+ builtin_scope.declare_struct_or_union(
417
+ name, "struct", scope, 1, None, cname = cname)
418
+
419
+
420
+ def init_builtins():
421
+ init_builtin_structs()
422
+ init_builtin_types()
423
+ init_builtin_funcs()
424
+
425
+ builtin_scope.declare_var(
426
+ '__debug__', PyrexTypes.c_const_type(PyrexTypes.c_bint_type),
427
+ pos=None, cname='(!Py_OptimizeFlag)', is_cdef=True)
428
+
429
+ global list_type, tuple_type, dict_type, set_type, frozenset_type
430
+ global bytes_type, str_type, unicode_type, basestring_type, slice_type
431
+ global float_type, bool_type, type_type, complex_type, bytearray_type
432
+ type_type = builtin_scope.lookup('type').type
433
+ list_type = builtin_scope.lookup('list').type
434
+ tuple_type = builtin_scope.lookup('tuple').type
435
+ dict_type = builtin_scope.lookup('dict').type
436
+ set_type = builtin_scope.lookup('set').type
437
+ frozenset_type = builtin_scope.lookup('frozenset').type
438
+ slice_type = builtin_scope.lookup('slice').type
439
+ bytes_type = builtin_scope.lookup('bytes').type
440
+ str_type = builtin_scope.lookup('str').type
441
+ unicode_type = builtin_scope.lookup('unicode').type
442
+ basestring_type = builtin_scope.lookup('basestring').type
443
+ bytearray_type = builtin_scope.lookup('bytearray').type
444
+ float_type = builtin_scope.lookup('float').type
445
+ bool_type = builtin_scope.lookup('bool').type
446
+ complex_type = builtin_scope.lookup('complex').type
447
+
448
+
449
+ init_builtins()
Cython/Compiler/CmdLine.py ADDED
@@ -0,0 +1,224 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #
2
+ # Cython - Command Line Parsing
3
+ #
4
+
5
+ from __future__ import absolute_import
6
+
7
+ import os
8
+ import sys
9
+ from . import Options
10
+
11
+ usage = """\
12
+ Cython (http://cython.org) is a compiler for code written in the
13
+ Cython language. Cython is based on Pyrex by Greg Ewing.
14
+
15
+ Usage: cython [options] sourcefile.{pyx,py} ...
16
+
17
+ Options:
18
+ -V, --version Display version number of cython compiler
19
+ -l, --create-listing Write error messages to a listing file
20
+ -I, --include-dir <directory> Search for include files in named directory
21
+ (multiple include directories are allowed).
22
+ -o, --output-file <filename> Specify name of generated C file
23
+ -t, --timestamps Only compile newer source files
24
+ -f, --force Compile all source files (overrides implied -t)
25
+ -v, --verbose Be verbose, print file names on multiple compilation
26
+ -p, --embed-positions If specified, the positions in Cython files of each
27
+ function definition is embedded in its docstring.
28
+ --cleanup <level> Release interned objects on python exit, for memory debugging.
29
+ Level indicates aggressiveness, default 0 releases nothing.
30
+ -w, --working <directory> Sets the working directory for Cython (the directory modules
31
+ are searched from)
32
+ --gdb Output debug information for cygdb
33
+ --gdb-outdir <directory> Specify gdb debug information output directory. Implies --gdb.
34
+
35
+ -D, --no-docstrings Strip docstrings from the compiled module.
36
+ -a, --annotate Produce a colorized HTML version of the source.
37
+ --annotate-coverage <cov.xml> Annotate and include coverage information from cov.xml.
38
+ --line-directives Produce #line directives pointing to the .pyx source
39
+ --cplus Output a C++ rather than C file.
40
+ --embed[=<method_name>] Generate a main() function that embeds the Python interpreter.
41
+ -2 Compile based on Python-2 syntax and code semantics.
42
+ -3 Compile based on Python-3 syntax and code semantics.
43
+ --3str Compile based on Python-3 syntax and code semantics without
44
+ assuming unicode by default for string literals under Python 2.
45
+ --lenient Change some compile time errors to runtime errors to
46
+ improve Python compatibility
47
+ --capi-reexport-cincludes Add cincluded headers to any auto-generated header files.
48
+ --fast-fail Abort the compilation on the first error
49
+ --warning-errors, -Werror Make all warnings into errors
50
+ --warning-extra, -Wextra Enable extra warnings
51
+ -X, --directive <name>=<value>[,<name=value,...] Overrides a compiler directive
52
+ -E, --compile-time-env name=value[,<name=value,...] Provides compile time env like DEF would do.
53
+ """
54
+
55
+
56
+ # The following experimental options are supported only on MacOSX:
57
+ # -C, --compile Compile generated .c file to .o file
58
+ # --link Link .o file to produce extension module (implies -C)
59
+ # -+, --cplus Use C++ compiler for compiling and linking
60
+ # Additional .o files to link may be supplied when using -X."""
61
+
62
+ def bad_usage():
63
+ sys.stderr.write(usage)
64
+ sys.exit(1)
65
+
66
+
67
+ def parse_command_line(args):
68
+ from .Main import CompilationOptions, default_options
69
+
70
+ pending_arg = []
71
+
72
+ def pop_arg():
73
+ if not args or pending_arg:
74
+ bad_usage()
75
+ if '=' in args[0] and args[0].startswith('--'): # allow "--long-option=xyz"
76
+ name, value = args.pop(0).split('=', 1)
77
+ pending_arg.append(value)
78
+ return name
79
+ return args.pop(0)
80
+
81
+ def pop_value(default=None):
82
+ if pending_arg:
83
+ return pending_arg.pop()
84
+ elif default is not None:
85
+ return default
86
+ elif not args:
87
+ bad_usage()
88
+ return args.pop(0)
89
+
90
+ def get_param(option):
91
+ tail = option[2:]
92
+ if tail:
93
+ return tail
94
+ else:
95
+ return pop_arg()
96
+
97
+ options = CompilationOptions(default_options)
98
+ sources = []
99
+ while args:
100
+ if args[0].startswith("-"):
101
+ option = pop_arg()
102
+ if option in ("-V", "--version"):
103
+ options.show_version = 1
104
+ elif option in ("-l", "--create-listing"):
105
+ options.use_listing_file = 1
106
+ elif option in ("-+", "--cplus"):
107
+ options.cplus = 1
108
+ elif option == "--embed":
109
+ Options.embed = pop_value("main")
110
+ elif option.startswith("-I"):
111
+ options.include_path.append(get_param(option))
112
+ elif option == "--include-dir":
113
+ options.include_path.append(pop_value())
114
+ elif option in ("-w", "--working"):
115
+ options.working_path = pop_value()
116
+ elif option in ("-o", "--output-file"):
117
+ options.output_file = pop_value()
118
+ elif option in ("-t", "--timestamps"):
119
+ options.timestamps = 1
120
+ elif option in ("-f", "--force"):
121
+ options.timestamps = 0
122
+ elif option in ("-v", "--verbose"):
123
+ options.verbose += 1
124
+ elif option in ("-p", "--embed-positions"):
125
+ Options.embed_pos_in_docstring = 1
126
+ elif option in ("-z", "--pre-import"):
127
+ Options.pre_import = pop_value()
128
+ elif option == "--cleanup":
129
+ Options.generate_cleanup_code = int(pop_value())
130
+ elif option in ("-D", "--no-docstrings"):
131
+ Options.docstrings = False
132
+ elif option in ("-a", "--annotate"):
133
+ Options.annotate = True
134
+ elif option == "--annotate-coverage":
135
+ Options.annotate = True
136
+ Options.annotate_coverage_xml = pop_value()
137
+ elif option == "--convert-range":
138
+ Options.convert_range = True
139
+ elif option == "--line-directives":
140
+ options.emit_linenums = True
141
+ elif option == "--no-c-in-traceback":
142
+ options.c_line_in_traceback = False
143
+ elif option == "--gdb":
144
+ options.gdb_debug = True
145
+ options.output_dir = os.curdir
146
+ elif option == "--gdb-outdir":
147
+ options.gdb_debug = True
148
+ options.output_dir = pop_value()
149
+ elif option == "--lenient":
150
+ Options.error_on_unknown_names = False
151
+ Options.error_on_uninitialized = False
152
+ elif option == '-2':
153
+ options.language_level = 2
154
+ elif option == '-3':
155
+ options.language_level = 3
156
+ elif option == '--3str':
157
+ options.language_level = '3str'
158
+ elif option == "--capi-reexport-cincludes":
159
+ options.capi_reexport_cincludes = True
160
+ elif option == "--fast-fail":
161
+ Options.fast_fail = True
162
+ elif option == "--cimport-from-pyx":
163
+ Options.cimport_from_pyx = True
164
+ elif option in ('-Werror', '--warning-errors'):
165
+ Options.warning_errors = True
166
+ elif option in ('-Wextra', '--warning-extra'):
167
+ options.compiler_directives.update(Options.extra_warnings)
168
+ elif option == "--old-style-globals":
169
+ Options.old_style_globals = True
170
+ elif option == "--directive" or option.startswith('-X'):
171
+ if option.startswith('-X') and option[2:].strip():
172
+ x_args = option[2:]
173
+ else:
174
+ x_args = pop_value()
175
+ try:
176
+ options.compiler_directives = Options.parse_directive_list(
177
+ x_args, relaxed_bool=True,
178
+ current_settings=options.compiler_directives)
179
+ except ValueError as e:
180
+ sys.stderr.write("Error in compiler directive: %s\n" % e.args[0])
181
+ sys.exit(1)
182
+ elif option == "--compile-time-env" or option.startswith('-E'):
183
+ if option.startswith('-E') and option[2:].strip():
184
+ x_args = option[2:]
185
+ else:
186
+ x_args = pop_value()
187
+ try:
188
+ options.compile_time_env = Options.parse_compile_time_env(
189
+ x_args, current_settings=options.compile_time_env)
190
+ except ValueError as e:
191
+ sys.stderr.write("Error in compile-time-env: %s\n" % e.args[0])
192
+ sys.exit(1)
193
+ elif option.startswith('--debug'):
194
+ option = option[2:].replace('-', '_')
195
+ from . import DebugFlags
196
+ if option in dir(DebugFlags):
197
+ setattr(DebugFlags, option, True)
198
+ else:
199
+ sys.stderr.write("Unknown debug flag: %s\n" % option)
200
+ bad_usage()
201
+ elif option in ('-h', '--help'):
202
+ sys.stdout.write(usage)
203
+ sys.exit(0)
204
+ else:
205
+ sys.stderr.write("Unknown compiler flag: %s\n" % option)
206
+ sys.exit(1)
207
+ else:
208
+ sources.append(pop_arg())
209
+
210
+ if pending_arg:
211
+ bad_usage()
212
+
213
+ if options.use_listing_file and len(sources) > 1:
214
+ sys.stderr.write(
215
+ "cython: Only one source file allowed when using -o\n")
216
+ sys.exit(1)
217
+ if len(sources) == 0 and not options.show_version:
218
+ bad_usage()
219
+ if Options.embed and len(sources) > 1:
220
+ sys.stderr.write(
221
+ "cython: Only one source file allowed when using -embed\n")
222
+ sys.exit(1)
223
+ return options, sources
224
+
Cython/Compiler/Code.pxd ADDED
@@ -0,0 +1,124 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+
2
+ from __future__ import absolute_import
3
+
4
+ cimport cython
5
+ from ..StringIOTree cimport StringIOTree
6
+
7
+
8
+ cdef class UtilityCodeBase(object):
9
+ cpdef format_code(self, code_string, replace_empty_lines=*)
10
+
11
+
12
+ cdef class UtilityCode(UtilityCodeBase):
13
+ cdef public object name
14
+ cdef public object proto
15
+ cdef public object impl
16
+ cdef public object init
17
+ cdef public object cleanup
18
+ cdef public object proto_block
19
+ cdef public object requires
20
+ cdef public dict _cache
21
+ cdef public list specialize_list
22
+ cdef public object file
23
+
24
+ cpdef none_or_sub(self, s, context)
25
+
26
+
27
+ cdef class FunctionState:
28
+ cdef public set names_taken
29
+ cdef public object owner
30
+ cdef public object scope
31
+
32
+ cdef public object error_label
33
+ cdef public size_t label_counter
34
+ cdef public set labels_used
35
+ cdef public object return_label
36
+ cdef public object continue_label
37
+ cdef public object break_label
38
+ cdef public list yield_labels
39
+
40
+ cdef public object return_from_error_cleanup_label # not used in __init__ ?
41
+
42
+ cdef public object exc_vars
43
+ cdef public object current_except
44
+ cdef public bint in_try_finally
45
+ cdef public bint can_trace
46
+ cdef public bint gil_owned
47
+
48
+ cdef public list temps_allocated
49
+ cdef public dict temps_free
50
+ cdef public dict temps_used_type
51
+ cdef public set zombie_temps
52
+ cdef public size_t temp_counter
53
+ cdef public list collect_temps_stack
54
+
55
+ cdef public object closure_temps
56
+ cdef public bint should_declare_error_indicator
57
+ cdef public bint uses_error_indicator
58
+
59
+ @cython.locals(n=size_t)
60
+ cpdef new_label(self, name=*)
61
+ cpdef tuple get_loop_labels(self)
62
+ cpdef set_loop_labels(self, labels)
63
+ cpdef tuple get_all_labels(self)
64
+ cpdef set_all_labels(self, labels)
65
+ cpdef start_collecting_temps(self)
66
+ cpdef stop_collecting_temps(self)
67
+
68
+ cpdef list temps_in_use(self)
69
+
70
+ cdef class IntConst:
71
+ cdef public object cname
72
+ cdef public object value
73
+ cdef public bint is_long
74
+
75
+ cdef class PyObjectConst:
76
+ cdef public object cname
77
+ cdef public object type
78
+
79
+ cdef class StringConst:
80
+ cdef public object cname
81
+ cdef public object text
82
+ cdef public object escaped_value
83
+ cdef public dict py_strings
84
+ cdef public list py_versions
85
+
86
+ @cython.locals(intern=bint, is_str=bint, is_unicode=bint)
87
+ cpdef get_py_string_const(self, encoding, identifier=*, is_str=*, py3str_cstring=*)
88
+
89
+ ## cdef class PyStringConst:
90
+ ## cdef public object cname
91
+ ## cdef public object encoding
92
+ ## cdef public bint is_str
93
+ ## cdef public bint is_unicode
94
+ ## cdef public bint intern
95
+
96
+ #class GlobalState(object):
97
+
98
+ #def funccontext_property(name):
99
+
100
+ cdef class CCodeWriter(object):
101
+ cdef readonly StringIOTree buffer
102
+ cdef readonly list pyclass_stack
103
+ cdef readonly object globalstate
104
+ cdef readonly object funcstate
105
+ cdef object code_config
106
+ cdef object last_pos
107
+ cdef object last_marked_pos
108
+ cdef Py_ssize_t level
109
+ cdef public Py_ssize_t call_level # debug-only, see Nodes.py
110
+ cdef bint bol
111
+
112
+ cpdef write(self, s)
113
+ cpdef put(self, code)
114
+ cpdef put_safe(self, code)
115
+ cpdef putln(self, code=*, bint safe=*)
116
+ @cython.final
117
+ cdef increase_indent(self)
118
+ @cython.final
119
+ cdef decrease_indent(self)
120
+
121
+
122
+ cdef class PyrexCodeWriter:
123
+ cdef public object f
124
+ cdef public Py_ssize_t level
Cython/Compiler/Code.py ADDED
@@ -0,0 +1,2597 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # cython: language_level = 2
2
+ # cython: auto_pickle=False
3
+ #
4
+ # Code output module
5
+ #
6
+
7
+ from __future__ import absolute_import
8
+
9
+ import cython
10
+ cython.declare(os=object, re=object, operator=object, textwrap=object,
11
+ Template=object, Naming=object, Options=object, StringEncoding=object,
12
+ Utils=object, SourceDescriptor=object, StringIOTree=object,
13
+ DebugFlags=object, basestring=object, defaultdict=object,
14
+ closing=object, partial=object)
15
+
16
+ import os
17
+ import re
18
+ import shutil
19
+ import sys
20
+ import operator
21
+ import textwrap
22
+ from string import Template
23
+ from functools import partial
24
+ from contextlib import closing
25
+ from collections import defaultdict
26
+
27
+ try:
28
+ import hashlib
29
+ except ImportError:
30
+ import md5 as hashlib
31
+
32
+ from . import Naming
33
+ from . import Options
34
+ from . import DebugFlags
35
+ from . import StringEncoding
36
+ from . import Version
37
+ from .. import Utils
38
+ from .Scanning import SourceDescriptor
39
+ from ..StringIOTree import StringIOTree
40
+
41
+ try:
42
+ from __builtin__ import basestring
43
+ except ImportError:
44
+ from builtins import str as basestring
45
+
46
+ KEYWORDS_MUST_BE_BYTES = sys.version_info < (2, 7)
47
+
48
+
49
+ non_portable_builtins_map = {
50
+ # builtins that have different names in different Python versions
51
+ 'bytes' : ('PY_MAJOR_VERSION < 3', 'str'),
52
+ 'unicode' : ('PY_MAJOR_VERSION >= 3', 'str'),
53
+ 'basestring' : ('PY_MAJOR_VERSION >= 3', 'str'),
54
+ 'xrange' : ('PY_MAJOR_VERSION >= 3', 'range'),
55
+ 'raw_input' : ('PY_MAJOR_VERSION >= 3', 'input'),
56
+ }
57
+
58
+ ctypedef_builtins_map = {
59
+ # types of builtins in "ctypedef class" statements which we don't
60
+ # import either because the names conflict with C types or because
61
+ # the type simply is not exposed.
62
+ 'py_int' : '&PyInt_Type',
63
+ 'py_long' : '&PyLong_Type',
64
+ 'py_float' : '&PyFloat_Type',
65
+ 'wrapper_descriptor' : '&PyWrapperDescr_Type',
66
+ }
67
+
68
+ basicsize_builtins_map = {
69
+ # builtins whose type has a different tp_basicsize than sizeof(...)
70
+ 'PyTypeObject': 'PyHeapTypeObject',
71
+ }
72
+
73
+ uncachable_builtins = [
74
+ # Global/builtin names that cannot be cached because they may or may not
75
+ # be available at import time, for various reasons:
76
+ ## - Py3.7+
77
+ 'breakpoint', # might deserve an implementation in Cython
78
+ ## - Py3.4+
79
+ '__loader__',
80
+ '__spec__',
81
+ ## - Py3+
82
+ 'BlockingIOError',
83
+ 'BrokenPipeError',
84
+ 'ChildProcessError',
85
+ 'ConnectionAbortedError',
86
+ 'ConnectionError',
87
+ 'ConnectionRefusedError',
88
+ 'ConnectionResetError',
89
+ 'FileExistsError',
90
+ 'FileNotFoundError',
91
+ 'InterruptedError',
92
+ 'IsADirectoryError',
93
+ 'ModuleNotFoundError',
94
+ 'NotADirectoryError',
95
+ 'PermissionError',
96
+ 'ProcessLookupError',
97
+ 'RecursionError',
98
+ 'ResourceWarning',
99
+ #'StopAsyncIteration', # backported
100
+ 'TimeoutError',
101
+ '__build_class__',
102
+ 'ascii', # might deserve an implementation in Cython
103
+ #'exec', # implemented in Cython
104
+ ## - Py2.7+
105
+ 'memoryview',
106
+ ## - platform specific
107
+ 'WindowsError',
108
+ ## - others
109
+ '_', # e.g. used by gettext
110
+ ]
111
+
112
+ special_py_methods = set([
113
+ '__cinit__', '__dealloc__', '__richcmp__', '__next__',
114
+ '__await__', '__aiter__', '__anext__',
115
+ '__getreadbuffer__', '__getwritebuffer__', '__getsegcount__',
116
+ '__getcharbuffer__', '__getbuffer__', '__releasebuffer__'
117
+ ])
118
+
119
+ modifier_output_mapper = {
120
+ 'inline': 'CYTHON_INLINE'
121
+ }.get
122
+
123
+
124
+ class IncludeCode(object):
125
+ """
126
+ An include file and/or verbatim C code to be included in the
127
+ generated sources.
128
+ """
129
+ # attributes:
130
+ #
131
+ # pieces {order: unicode}: pieces of C code to be generated.
132
+ # For the included file, the key "order" is zero.
133
+ # For verbatim include code, the "order" is the "order"
134
+ # attribute of the original IncludeCode where this piece
135
+ # of C code was first added. This is needed to prevent
136
+ # duplication if the same include code is found through
137
+ # multiple cimports.
138
+ # location int: where to put this include in the C sources, one
139
+ # of the constants INITIAL, EARLY, LATE
140
+ # order int: sorting order (automatically set by increasing counter)
141
+
142
+ # Constants for location. If the same include occurs with different
143
+ # locations, the earliest one takes precedense.
144
+ INITIAL = 0
145
+ EARLY = 1
146
+ LATE = 2
147
+
148
+ counter = 1 # Counter for "order"
149
+
150
+ def __init__(self, include=None, verbatim=None, late=True, initial=False):
151
+ self.order = self.counter
152
+ type(self).counter += 1
153
+ self.pieces = {}
154
+
155
+ if include:
156
+ if include[0] == '<' and include[-1] == '>':
157
+ self.pieces[0] = u'#include {0}'.format(include)
158
+ late = False # system include is never late
159
+ else:
160
+ self.pieces[0] = u'#include "{0}"'.format(include)
161
+
162
+ if verbatim:
163
+ self.pieces[self.order] = verbatim
164
+
165
+ if initial:
166
+ self.location = self.INITIAL
167
+ elif late:
168
+ self.location = self.LATE
169
+ else:
170
+ self.location = self.EARLY
171
+
172
+ def dict_update(self, d, key):
173
+ """
174
+ Insert `self` in dict `d` with key `key`. If that key already
175
+ exists, update the attributes of the existing value with `self`.
176
+ """
177
+ if key in d:
178
+ other = d[key]
179
+ other.location = min(self.location, other.location)
180
+ other.pieces.update(self.pieces)
181
+ else:
182
+ d[key] = self
183
+
184
+ def sortkey(self):
185
+ return self.order
186
+
187
+ def mainpiece(self):
188
+ """
189
+ Return the main piece of C code, corresponding to the include
190
+ file. If there was no include file, return None.
191
+ """
192
+ return self.pieces.get(0)
193
+
194
+ def write(self, code):
195
+ # Write values of self.pieces dict, sorted by the keys
196
+ for k in sorted(self.pieces):
197
+ code.putln(self.pieces[k])
198
+
199
+
200
+ def get_utility_dir():
201
+ # make this a function and not global variables:
202
+ # http://trac.cython.org/cython_trac/ticket/475
203
+ Cython_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
204
+ return os.path.join(Cython_dir, "Utility")
205
+
206
+
207
+ class UtilityCodeBase(object):
208
+ """
209
+ Support for loading utility code from a file.
210
+
211
+ Code sections in the file can be specified as follows:
212
+
213
+ ##### MyUtility.proto #####
214
+
215
+ [proto declarations]
216
+
217
+ ##### MyUtility.init #####
218
+
219
+ [code run at module initialization]
220
+
221
+ ##### MyUtility #####
222
+ #@requires: MyOtherUtility
223
+ #@substitute: naming
224
+
225
+ [definitions]
226
+
227
+ for prototypes and implementation respectively. For non-python or
228
+ -cython files backslashes should be used instead. 5 to 30 comment
229
+ characters may be used on either side.
230
+
231
+ If the @cname decorator is not used and this is a CythonUtilityCode,
232
+ one should pass in the 'name' keyword argument to be used for name
233
+ mangling of such entries.
234
+ """
235
+
236
+ is_cython_utility = False
237
+ _utility_cache = {}
238
+
239
+ @classmethod
240
+ def _add_utility(cls, utility, type, lines, begin_lineno, tags=None):
241
+ if utility is None:
242
+ return
243
+
244
+ code = '\n'.join(lines)
245
+ if tags and 'substitute' in tags and tags['substitute'] == set(['naming']):
246
+ del tags['substitute']
247
+ try:
248
+ code = Template(code).substitute(vars(Naming))
249
+ except (KeyError, ValueError) as e:
250
+ raise RuntimeError("Error parsing templated utility code of type '%s' at line %d: %s" % (
251
+ type, begin_lineno, e))
252
+
253
+ # remember correct line numbers at least until after templating
254
+ code = '\n' * begin_lineno + code
255
+
256
+ if type == 'proto':
257
+ utility[0] = code
258
+ elif type == 'impl':
259
+ utility[1] = code
260
+ else:
261
+ all_tags = utility[2]
262
+ if KEYWORDS_MUST_BE_BYTES:
263
+ type = type.encode('ASCII')
264
+ all_tags[type] = code
265
+
266
+ if tags:
267
+ all_tags = utility[2]
268
+ for name, values in tags.items():
269
+ if KEYWORDS_MUST_BE_BYTES:
270
+ name = name.encode('ASCII')
271
+ all_tags.setdefault(name, set()).update(values)
272
+
273
+ @classmethod
274
+ def load_utilities_from_file(cls, path):
275
+ utilities = cls._utility_cache.get(path)
276
+ if utilities:
277
+ return utilities
278
+
279
+ filename = os.path.join(get_utility_dir(), path)
280
+ _, ext = os.path.splitext(path)
281
+ if ext in ('.pyx', '.py', '.pxd', '.pxi'):
282
+ comment = '#'
283
+ strip_comments = partial(re.compile(r'^\s*#(?!\s*cython\s*:).*').sub, '')
284
+ rstrip = StringEncoding._unicode.rstrip
285
+ else:
286
+ comment = '/'
287
+ strip_comments = partial(re.compile(r'^\s*//.*|/\*[^*]*\*/').sub, '')
288
+ rstrip = partial(re.compile(r'\s+(\\?)$').sub, r'\1')
289
+ match_special = re.compile(
290
+ (r'^%(C)s{5,30}\s*(?P<name>(?:\w|\.)+)\s*%(C)s{5,30}|'
291
+ r'^%(C)s+@(?P<tag>\w+)\s*:\s*(?P<value>(?:\w|[.:])+)') %
292
+ {'C': comment}).match
293
+ match_type = re.compile(r'(.+)[.](proto(?:[.]\S+)?|impl|init|cleanup)$').match
294
+
295
+ with closing(Utils.open_source_file(filename, encoding='UTF-8')) as f:
296
+ all_lines = f.readlines()
297
+
298
+ utilities = defaultdict(lambda: [None, None, {}])
299
+ lines = []
300
+ tags = defaultdict(set)
301
+ utility = type = None
302
+ begin_lineno = 0
303
+
304
+ for lineno, line in enumerate(all_lines):
305
+ m = match_special(line)
306
+ if m:
307
+ if m.group('name'):
308
+ cls._add_utility(utility, type, lines, begin_lineno, tags)
309
+
310
+ begin_lineno = lineno + 1
311
+ del lines[:]
312
+ tags.clear()
313
+
314
+ name = m.group('name')
315
+ mtype = match_type(name)
316
+ if mtype:
317
+ name, type = mtype.groups()
318
+ else:
319
+ type = 'impl'
320
+ utility = utilities[name]
321
+ else:
322
+ tags[m.group('tag')].add(m.group('value'))
323
+ lines.append('') # keep line number correct
324
+ else:
325
+ lines.append(rstrip(strip_comments(line)))
326
+
327
+ if utility is None:
328
+ raise ValueError("Empty utility code file")
329
+
330
+ # Don't forget to add the last utility code
331
+ cls._add_utility(utility, type, lines, begin_lineno, tags)
332
+
333
+ utilities = dict(utilities) # un-defaultdict-ify
334
+ cls._utility_cache[path] = utilities
335
+ return utilities
336
+
337
+ @classmethod
338
+ def load(cls, util_code_name, from_file=None, **kwargs):
339
+ """
340
+ Load utility code from a file specified by from_file (relative to
341
+ Cython/Utility) and name util_code_name. If from_file is not given,
342
+ load it from the file util_code_name.*. There should be only one
343
+ file matched by this pattern.
344
+ """
345
+ if '::' in util_code_name:
346
+ from_file, util_code_name = util_code_name.rsplit('::', 1)
347
+ if not from_file:
348
+ utility_dir = get_utility_dir()
349
+ prefix = util_code_name + '.'
350
+ try:
351
+ listing = os.listdir(utility_dir)
352
+ except OSError:
353
+ # XXX the code below assumes as 'zipimport.zipimporter' instance
354
+ # XXX should be easy to generalize, but too lazy right now to write it
355
+ import zipfile
356
+ global __loader__
357
+ loader = __loader__
358
+ archive = loader.archive
359
+ with closing(zipfile.ZipFile(archive)) as fileobj:
360
+ listing = [os.path.basename(name)
361
+ for name in fileobj.namelist()
362
+ if os.path.join(archive, name).startswith(utility_dir)]
363
+ files = [filename for filename in listing
364
+ if filename.startswith(prefix)]
365
+ if not files:
366
+ raise ValueError("No match found for utility code " + util_code_name)
367
+ if len(files) > 1:
368
+ raise ValueError("More than one filename match found for utility code " + util_code_name)
369
+ from_file = files[0]
370
+
371
+ utilities = cls.load_utilities_from_file(from_file)
372
+ proto, impl, tags = utilities[util_code_name]
373
+
374
+ if tags:
375
+ orig_kwargs = kwargs.copy()
376
+ for name, values in tags.items():
377
+ if name in kwargs:
378
+ continue
379
+ # only pass lists when we have to: most argument expect one value or None
380
+ if name == 'requires':
381
+ if orig_kwargs:
382
+ values = [cls.load(dep, from_file, **orig_kwargs)
383
+ for dep in sorted(values)]
384
+ else:
385
+ # dependencies are rarely unique, so use load_cached() when we can
386
+ values = [cls.load_cached(dep, from_file)
387
+ for dep in sorted(values)]
388
+ elif not values:
389
+ values = None
390
+ elif len(values) == 1:
391
+ values = list(values)[0]
392
+ kwargs[name] = values
393
+
394
+ if proto is not None:
395
+ kwargs['proto'] = proto
396
+ if impl is not None:
397
+ kwargs['impl'] = impl
398
+
399
+ if 'name' not in kwargs:
400
+ kwargs['name'] = util_code_name
401
+
402
+ if 'file' not in kwargs and from_file:
403
+ kwargs['file'] = from_file
404
+ return cls(**kwargs)
405
+
406
+ @classmethod
407
+ def load_cached(cls, utility_code_name, from_file=None, __cache={}):
408
+ """
409
+ Calls .load(), but using a per-type cache based on utility name and file name.
410
+ """
411
+ key = (cls, from_file, utility_code_name)
412
+ try:
413
+ return __cache[key]
414
+ except KeyError:
415
+ pass
416
+ code = __cache[key] = cls.load(utility_code_name, from_file)
417
+ return code
418
+
419
+ @classmethod
420
+ def load_as_string(cls, util_code_name, from_file=None, **kwargs):
421
+ """
422
+ Load a utility code as a string. Returns (proto, implementation)
423
+ """
424
+ util = cls.load(util_code_name, from_file, **kwargs)
425
+ proto, impl = util.proto, util.impl
426
+ return util.format_code(proto), util.format_code(impl)
427
+
428
+ def format_code(self, code_string, replace_empty_lines=re.compile(r'\n\n+').sub):
429
+ """
430
+ Format a code section for output.
431
+ """
432
+ if code_string:
433
+ code_string = replace_empty_lines('\n', code_string.strip()) + '\n\n'
434
+ return code_string
435
+
436
+ def __str__(self):
437
+ return "<%s(%s)>" % (type(self).__name__, self.name)
438
+
439
+ def get_tree(self, **kwargs):
440
+ pass
441
+
442
+ def __deepcopy__(self, memodict=None):
443
+ # No need to deep-copy utility code since it's essentially immutable.
444
+ return self
445
+
446
+
447
+ class UtilityCode(UtilityCodeBase):
448
+ """
449
+ Stores utility code to add during code generation.
450
+
451
+ See GlobalState.put_utility_code.
452
+
453
+ hashes/equals by instance
454
+
455
+ proto C prototypes
456
+ impl implementation code
457
+ init code to call on module initialization
458
+ requires utility code dependencies
459
+ proto_block the place in the resulting file where the prototype should
460
+ end up
461
+ name name of the utility code (or None)
462
+ file filename of the utility code file this utility was loaded
463
+ from (or None)
464
+ """
465
+
466
+ def __init__(self, proto=None, impl=None, init=None, cleanup=None, requires=None,
467
+ proto_block='utility_code_proto', name=None, file=None):
468
+ # proto_block: Which code block to dump prototype in. See GlobalState.
469
+ self.proto = proto
470
+ self.impl = impl
471
+ self.init = init
472
+ self.cleanup = cleanup
473
+ self.requires = requires
474
+ self._cache = {}
475
+ self.specialize_list = []
476
+ self.proto_block = proto_block
477
+ self.name = name
478
+ self.file = file
479
+
480
+ def __hash__(self):
481
+ return hash((self.proto, self.impl))
482
+
483
+ def __eq__(self, other):
484
+ if self is other:
485
+ return True
486
+ self_type, other_type = type(self), type(other)
487
+ if self_type is not other_type and not (isinstance(other, self_type) or isinstance(self, other_type)):
488
+ return False
489
+
490
+ self_proto = getattr(self, 'proto', None)
491
+ other_proto = getattr(other, 'proto', None)
492
+ return (self_proto, self.impl) == (other_proto, other.impl)
493
+
494
+ def none_or_sub(self, s, context):
495
+ """
496
+ Format a string in this utility code with context. If None, do nothing.
497
+ """
498
+ if s is None:
499
+ return None
500
+ return s % context
501
+
502
+ def specialize(self, pyrex_type=None, **data):
503
+ # Dicts aren't hashable...
504
+ name = self.name
505
+ if pyrex_type is not None:
506
+ data['type'] = pyrex_type.empty_declaration_code()
507
+ data['type_name'] = pyrex_type.specialization_name()
508
+ name = "%s[%s]" % (name, data['type_name'])
509
+ key = tuple(sorted(data.items()))
510
+ try:
511
+ return self._cache[key]
512
+ except KeyError:
513
+ if self.requires is None:
514
+ requires = None
515
+ else:
516
+ requires = [r.specialize(data) for r in self.requires]
517
+
518
+ s = self._cache[key] = UtilityCode(
519
+ self.none_or_sub(self.proto, data),
520
+ self.none_or_sub(self.impl, data),
521
+ self.none_or_sub(self.init, data),
522
+ self.none_or_sub(self.cleanup, data),
523
+ requires,
524
+ self.proto_block,
525
+ name,
526
+ )
527
+
528
+ self.specialize_list.append(s)
529
+ return s
530
+
531
+ def inject_string_constants(self, impl, output):
532
+ """Replace 'PYIDENT("xyz")' by a constant Python identifier cname.
533
+ """
534
+ if 'PYIDENT(' not in impl and 'PYUNICODE(' not in impl:
535
+ return False, impl
536
+
537
+ replacements = {}
538
+ def externalise(matchobj):
539
+ key = matchobj.groups()
540
+ try:
541
+ cname = replacements[key]
542
+ except KeyError:
543
+ str_type, name = key
544
+ cname = replacements[key] = output.get_py_string_const(
545
+ StringEncoding.EncodedString(name), identifier=str_type == 'IDENT').cname
546
+ return cname
547
+
548
+ impl = re.sub(r'PY(IDENT|UNICODE)\("([^"]+)"\)', externalise, impl)
549
+ assert 'PYIDENT(' not in impl and 'PYUNICODE(' not in impl
550
+ return True, impl
551
+
552
+ def inject_unbound_methods(self, impl, output):
553
+ """Replace 'UNBOUND_METHOD(type, "name")' by a constant Python identifier cname.
554
+ """
555
+ if 'CALL_UNBOUND_METHOD(' not in impl:
556
+ return False, impl
557
+
558
+ def externalise(matchobj):
559
+ type_cname, method_name, obj_cname, args = matchobj.groups()
560
+ args = [arg.strip() for arg in args[1:].split(',')] if args else []
561
+ assert len(args) < 3, "CALL_UNBOUND_METHOD() does not support %d call arguments" % len(args)
562
+ return output.cached_unbound_method_call_code(obj_cname, type_cname, method_name, args)
563
+
564
+ impl = re.sub(
565
+ r'CALL_UNBOUND_METHOD\('
566
+ r'([a-zA-Z_]+),' # type cname
567
+ r'\s*"([^"]+)",' # method name
568
+ r'\s*([^),]+)' # object cname
569
+ r'((?:,\s*[^),]+)*)' # args*
570
+ r'\)', externalise, impl)
571
+ assert 'CALL_UNBOUND_METHOD(' not in impl
572
+
573
+ return True, impl
574
+
575
+ def wrap_c_strings(self, impl):
576
+ """Replace CSTRING('''xyz''') by a C compatible string
577
+ """
578
+ if 'CSTRING(' not in impl:
579
+ return impl
580
+
581
+ def split_string(matchobj):
582
+ content = matchobj.group(1).replace('"', '\042')
583
+ return ''.join(
584
+ '"%s\\n"\n' % line if not line.endswith('\\') or line.endswith('\\\\') else '"%s"\n' % line[:-1]
585
+ for line in content.splitlines())
586
+
587
+ impl = re.sub(r'CSTRING\(\s*"""([^"]*(?:"[^"]+)*)"""\s*\)', split_string, impl)
588
+ assert 'CSTRING(' not in impl
589
+ return impl
590
+
591
+ def put_code(self, output):
592
+ if self.requires:
593
+ for dependency in self.requires:
594
+ output.use_utility_code(dependency)
595
+ if self.proto:
596
+ writer = output[self.proto_block]
597
+ writer.putln("/* %s.proto */" % self.name)
598
+ writer.put_or_include(
599
+ self.format_code(self.proto), '%s_proto' % self.name)
600
+ if self.impl:
601
+ impl = self.format_code(self.wrap_c_strings(self.impl))
602
+ is_specialised1, impl = self.inject_string_constants(impl, output)
603
+ is_specialised2, impl = self.inject_unbound_methods(impl, output)
604
+ writer = output['utility_code_def']
605
+ writer.putln("/* %s */" % self.name)
606
+ if not (is_specialised1 or is_specialised2):
607
+ # no module specific adaptations => can be reused
608
+ writer.put_or_include(impl, '%s_impl' % self.name)
609
+ else:
610
+ writer.put(impl)
611
+ if self.init:
612
+ writer = output['init_globals']
613
+ writer.putln("/* %s.init */" % self.name)
614
+ if isinstance(self.init, basestring):
615
+ writer.put(self.format_code(self.init))
616
+ else:
617
+ self.init(writer, output.module_pos)
618
+ writer.putln(writer.error_goto_if_PyErr(output.module_pos))
619
+ writer.putln()
620
+ if self.cleanup and Options.generate_cleanup_code:
621
+ writer = output['cleanup_globals']
622
+ writer.putln("/* %s.cleanup */" % self.name)
623
+ if isinstance(self.cleanup, basestring):
624
+ writer.put_or_include(
625
+ self.format_code(self.cleanup),
626
+ '%s_cleanup' % self.name)
627
+ else:
628
+ self.cleanup(writer, output.module_pos)
629
+
630
+
631
+ def sub_tempita(s, context, file=None, name=None):
632
+ "Run tempita on string s with given context."
633
+ if not s:
634
+ return None
635
+
636
+ if file:
637
+ context['__name'] = "%s:%s" % (file, name)
638
+ elif name:
639
+ context['__name'] = name
640
+
641
+ from ..Tempita import sub
642
+ return sub(s, **context)
643
+
644
+
645
+ class TempitaUtilityCode(UtilityCode):
646
+ def __init__(self, name=None, proto=None, impl=None, init=None, file=None, context=None, **kwargs):
647
+ if context is None:
648
+ context = {}
649
+ proto = sub_tempita(proto, context, file, name)
650
+ impl = sub_tempita(impl, context, file, name)
651
+ init = sub_tempita(init, context, file, name)
652
+ super(TempitaUtilityCode, self).__init__(
653
+ proto, impl, init=init, name=name, file=file, **kwargs)
654
+
655
+ @classmethod
656
+ def load_cached(cls, utility_code_name, from_file=None, context=None, __cache={}):
657
+ context_key = tuple(sorted(context.items())) if context else None
658
+ assert hash(context_key) is not None # raise TypeError if not hashable
659
+ key = (cls, from_file, utility_code_name, context_key)
660
+ try:
661
+ return __cache[key]
662
+ except KeyError:
663
+ pass
664
+ code = __cache[key] = cls.load(utility_code_name, from_file, context=context)
665
+ return code
666
+
667
+ def none_or_sub(self, s, context):
668
+ """
669
+ Format a string in this utility code with context. If None, do nothing.
670
+ """
671
+ if s is None:
672
+ return None
673
+ return sub_tempita(s, context, self.file, self.name)
674
+
675
+
676
+ class LazyUtilityCode(UtilityCodeBase):
677
+ """
678
+ Utility code that calls a callback with the root code writer when
679
+ available. Useful when you only have 'env' but not 'code'.
680
+ """
681
+ __name__ = '<lazy>'
682
+ requires = None
683
+
684
+ def __init__(self, callback):
685
+ self.callback = callback
686
+
687
+ def put_code(self, globalstate):
688
+ utility = self.callback(globalstate.rootwriter)
689
+ globalstate.use_utility_code(utility)
690
+
691
+
692
+ class FunctionState(object):
693
+ # return_label string function return point label
694
+ # error_label string error catch point label
695
+ # continue_label string loop continue point label
696
+ # break_label string loop break point label
697
+ # return_from_error_cleanup_label string
698
+ # label_counter integer counter for naming labels
699
+ # in_try_finally boolean inside try of try...finally
700
+ # exc_vars (string * 3) exception variables for reraise, or None
701
+ # can_trace boolean line tracing is supported in the current context
702
+ # scope Scope the scope object of the current function
703
+
704
+ # Not used for now, perhaps later
705
+ def __init__(self, owner, names_taken=set(), scope=None):
706
+ self.names_taken = names_taken
707
+ self.owner = owner
708
+ self.scope = scope
709
+
710
+ self.error_label = None
711
+ self.label_counter = 0
712
+ self.labels_used = set()
713
+ self.return_label = self.new_label()
714
+ self.new_error_label()
715
+ self.continue_label = None
716
+ self.break_label = None
717
+ self.yield_labels = []
718
+
719
+ self.in_try_finally = 0
720
+ self.exc_vars = None
721
+ self.current_except = None
722
+ self.can_trace = False
723
+ self.gil_owned = True
724
+
725
+ self.temps_allocated = [] # of (name, type, manage_ref, static)
726
+ self.temps_free = {} # (type, manage_ref) -> list of free vars with same type/managed status
727
+ self.temps_used_type = {} # name -> (type, manage_ref)
728
+ self.zombie_temps = set() # temps that must not be reused after release
729
+ self.temp_counter = 0
730
+ self.closure_temps = None
731
+
732
+ # This is used to collect temporaries, useful to find out which temps
733
+ # need to be privatized in parallel sections
734
+ self.collect_temps_stack = []
735
+
736
+ # This is used for the error indicator, which needs to be local to the
737
+ # function. It used to be global, which relies on the GIL being held.
738
+ # However, exceptions may need to be propagated through 'nogil'
739
+ # sections, in which case we introduce a race condition.
740
+ self.should_declare_error_indicator = False
741
+ self.uses_error_indicator = False
742
+
743
+ # safety checks
744
+
745
+ def validate_exit(self):
746
+ # validate that all allocated temps have been freed
747
+ if self.temps_allocated:
748
+ leftovers = self.temps_in_use()
749
+ if leftovers:
750
+ msg = "TEMPGUARD: Temps left over at end of '%s': %s" % (self.scope.name, ', '.join([
751
+ '%s [%s]' % (name, ctype)
752
+ for name, ctype, is_pytemp in sorted(leftovers)]),
753
+ )
754
+ #print(msg)
755
+ raise RuntimeError(msg)
756
+
757
+ # labels
758
+
759
+ def new_label(self, name=None):
760
+ n = self.label_counter
761
+ self.label_counter = n + 1
762
+ label = "%s%d" % (Naming.label_prefix, n)
763
+ if name is not None:
764
+ label += '_' + name
765
+ return label
766
+
767
+ def new_yield_label(self, expr_type='yield'):
768
+ label = self.new_label('resume_from_%s' % expr_type)
769
+ num_and_label = (len(self.yield_labels) + 1, label)
770
+ self.yield_labels.append(num_and_label)
771
+ return num_and_label
772
+
773
+ def new_error_label(self):
774
+ old_err_lbl = self.error_label
775
+ self.error_label = self.new_label('error')
776
+ return old_err_lbl
777
+
778
+ def get_loop_labels(self):
779
+ return (
780
+ self.continue_label,
781
+ self.break_label)
782
+
783
+ def set_loop_labels(self, labels):
784
+ (self.continue_label,
785
+ self.break_label) = labels
786
+
787
+ def new_loop_labels(self):
788
+ old_labels = self.get_loop_labels()
789
+ self.set_loop_labels(
790
+ (self.new_label("continue"),
791
+ self.new_label("break")))
792
+ return old_labels
793
+
794
+ def get_all_labels(self):
795
+ return (
796
+ self.continue_label,
797
+ self.break_label,
798
+ self.return_label,
799
+ self.error_label)
800
+
801
+ def set_all_labels(self, labels):
802
+ (self.continue_label,
803
+ self.break_label,
804
+ self.return_label,
805
+ self.error_label) = labels
806
+
807
+ def all_new_labels(self):
808
+ old_labels = self.get_all_labels()
809
+ new_labels = []
810
+ for old_label, name in zip(old_labels, ['continue', 'break', 'return', 'error']):
811
+ if old_label:
812
+ new_labels.append(self.new_label(name))
813
+ else:
814
+ new_labels.append(old_label)
815
+ self.set_all_labels(new_labels)
816
+ return old_labels
817
+
818
+ def use_label(self, lbl):
819
+ self.labels_used.add(lbl)
820
+
821
+ def label_used(self, lbl):
822
+ return lbl in self.labels_used
823
+
824
+ # temp handling
825
+
826
+ def allocate_temp(self, type, manage_ref, static=False, reusable=True):
827
+ """
828
+ Allocates a temporary (which may create a new one or get a previously
829
+ allocated and released one of the same type). Type is simply registered
830
+ and handed back, but will usually be a PyrexType.
831
+
832
+ If type.is_pyobject, manage_ref comes into play. If manage_ref is set to
833
+ True, the temp will be decref-ed on return statements and in exception
834
+ handling clauses. Otherwise the caller has to deal with any reference
835
+ counting of the variable.
836
+
837
+ If not type.is_pyobject, then manage_ref will be ignored, but it
838
+ still has to be passed. It is recommended to pass False by convention
839
+ if it is known that type will never be a Python object.
840
+
841
+ static=True marks the temporary declaration with "static".
842
+ This is only used when allocating backing store for a module-level
843
+ C array literals.
844
+
845
+ if reusable=False, the temp will not be reused after release.
846
+
847
+ A C string referring to the variable is returned.
848
+ """
849
+ if type.is_const and not type.is_reference:
850
+ type = type.const_base_type
851
+ elif type.is_reference and not type.is_fake_reference:
852
+ type = type.ref_base_type
853
+ elif type.is_cfunction:
854
+ from . import PyrexTypes
855
+ type = PyrexTypes.c_ptr_type(type) # A function itself isn't an l-value
856
+ if not type.is_pyobject and not type.is_memoryviewslice:
857
+ # Make manage_ref canonical, so that manage_ref will always mean
858
+ # a decref is needed.
859
+ manage_ref = False
860
+
861
+ freelist = self.temps_free.get((type, manage_ref))
862
+ if reusable and freelist is not None and freelist[0]:
863
+ result = freelist[0].pop()
864
+ freelist[1].remove(result)
865
+ else:
866
+ while True:
867
+ self.temp_counter += 1
868
+ result = "%s%d" % (Naming.codewriter_temp_prefix, self.temp_counter)
869
+ if result not in self.names_taken: break
870
+ self.temps_allocated.append((result, type, manage_ref, static))
871
+ if not reusable:
872
+ self.zombie_temps.add(result)
873
+ self.temps_used_type[result] = (type, manage_ref)
874
+ if DebugFlags.debug_temp_code_comments:
875
+ self.owner.putln("/* %s allocated (%s)%s */" % (result, type, "" if reusable else " - zombie"))
876
+
877
+ if self.collect_temps_stack:
878
+ self.collect_temps_stack[-1].add((result, type))
879
+
880
+ return result
881
+
882
+ def release_temp(self, name):
883
+ """
884
+ Releases a temporary so that it can be reused by other code needing
885
+ a temp of the same type.
886
+ """
887
+ type, manage_ref = self.temps_used_type[name]
888
+ freelist = self.temps_free.get((type, manage_ref))
889
+ if freelist is None:
890
+ freelist = ([], set()) # keep order in list and make lookups in set fast
891
+ self.temps_free[(type, manage_ref)] = freelist
892
+ if name in freelist[1]:
893
+ raise RuntimeError("Temp %s freed twice!" % name)
894
+ if name not in self.zombie_temps:
895
+ freelist[0].append(name)
896
+ freelist[1].add(name)
897
+ if DebugFlags.debug_temp_code_comments:
898
+ self.owner.putln("/* %s released %s*/" % (
899
+ name, " - zombie" if name in self.zombie_temps else ""))
900
+
901
+ def temps_in_use(self):
902
+ """Return a list of (cname,type,manage_ref) tuples of temp names and their type
903
+ that are currently in use.
904
+ """
905
+ used = []
906
+ for name, type, manage_ref, static in self.temps_allocated:
907
+ freelist = self.temps_free.get((type, manage_ref))
908
+ if freelist is None or name not in freelist[1]:
909
+ used.append((name, type, manage_ref and type.is_pyobject))
910
+ return used
911
+
912
+ def temps_holding_reference(self):
913
+ """Return a list of (cname,type) tuples of temp names and their type
914
+ that are currently in use. This includes only temps of a
915
+ Python object type which owns its reference.
916
+ """
917
+ return [(name, type)
918
+ for name, type, manage_ref in self.temps_in_use()
919
+ if manage_ref and type.is_pyobject]
920
+
921
+ def all_managed_temps(self):
922
+ """Return a list of (cname, type) tuples of refcount-managed Python objects.
923
+ """
924
+ return [(cname, type)
925
+ for cname, type, manage_ref, static in self.temps_allocated
926
+ if manage_ref]
927
+
928
+ def all_free_managed_temps(self):
929
+ """Return a list of (cname, type) tuples of refcount-managed Python
930
+ objects that are not currently in use. This is used by
931
+ try-except and try-finally blocks to clean up temps in the
932
+ error case.
933
+ """
934
+ return sorted([ # Enforce deterministic order.
935
+ (cname, type)
936
+ for (type, manage_ref), freelist in self.temps_free.items() if manage_ref
937
+ for cname in freelist[0]
938
+ ])
939
+
940
+ def start_collecting_temps(self):
941
+ """
942
+ Useful to find out which temps were used in a code block
943
+ """
944
+ self.collect_temps_stack.append(set())
945
+
946
+ def stop_collecting_temps(self):
947
+ return self.collect_temps_stack.pop()
948
+
949
+ def init_closure_temps(self, scope):
950
+ self.closure_temps = ClosureTempAllocator(scope)
951
+
952
+
953
+ class NumConst(object):
954
+ """Global info about a Python number constant held by GlobalState.
955
+
956
+ cname string
957
+ value string
958
+ py_type string int, long, float
959
+ value_code string evaluation code if different from value
960
+ """
961
+
962
+ def __init__(self, cname, value, py_type, value_code=None):
963
+ self.cname = cname
964
+ self.value = value
965
+ self.py_type = py_type
966
+ self.value_code = value_code or value
967
+
968
+
969
+ class PyObjectConst(object):
970
+ """Global info about a generic constant held by GlobalState.
971
+ """
972
+ # cname string
973
+ # type PyrexType
974
+
975
+ def __init__(self, cname, type):
976
+ self.cname = cname
977
+ self.type = type
978
+
979
+
980
+ cython.declare(possible_unicode_identifier=object, possible_bytes_identifier=object,
981
+ replace_identifier=object, find_alphanums=object)
982
+ possible_unicode_identifier = re.compile(br"(?![0-9])\w+$".decode('ascii'), re.U).match
983
+ possible_bytes_identifier = re.compile(r"(?![0-9])\w+$".encode('ASCII')).match
984
+ replace_identifier = re.compile(r'[^a-zA-Z0-9_]+').sub
985
+ find_alphanums = re.compile('([a-zA-Z0-9]+)').findall
986
+
987
+ class StringConst(object):
988
+ """Global info about a C string constant held by GlobalState.
989
+ """
990
+ # cname string
991
+ # text EncodedString or BytesLiteral
992
+ # py_strings {(identifier, encoding) : PyStringConst}
993
+
994
+ def __init__(self, cname, text, byte_string):
995
+ self.cname = cname
996
+ self.text = text
997
+ self.escaped_value = StringEncoding.escape_byte_string(byte_string)
998
+ self.py_strings = None
999
+ self.py_versions = []
1000
+
1001
+ def add_py_version(self, version):
1002
+ if not version:
1003
+ self.py_versions = [2, 3]
1004
+ elif version not in self.py_versions:
1005
+ self.py_versions.append(version)
1006
+
1007
+ def get_py_string_const(self, encoding, identifier=None,
1008
+ is_str=False, py3str_cstring=None):
1009
+ py_strings = self.py_strings
1010
+ text = self.text
1011
+
1012
+ is_str = bool(identifier or is_str)
1013
+ is_unicode = encoding is None and not is_str
1014
+
1015
+ if encoding is None:
1016
+ # unicode string
1017
+ encoding_key = None
1018
+ else:
1019
+ # bytes or str
1020
+ encoding = encoding.lower()
1021
+ if encoding in ('utf8', 'utf-8', 'ascii', 'usascii', 'us-ascii'):
1022
+ encoding = None
1023
+ encoding_key = None
1024
+ else:
1025
+ encoding_key = ''.join(find_alphanums(encoding))
1026
+
1027
+ key = (is_str, is_unicode, encoding_key, py3str_cstring)
1028
+ if py_strings is not None:
1029
+ try:
1030
+ return py_strings[key]
1031
+ except KeyError:
1032
+ pass
1033
+ else:
1034
+ self.py_strings = {}
1035
+
1036
+ if identifier:
1037
+ intern = True
1038
+ elif identifier is None:
1039
+ if isinstance(text, bytes):
1040
+ intern = bool(possible_bytes_identifier(text))
1041
+ else:
1042
+ intern = bool(possible_unicode_identifier(text))
1043
+ else:
1044
+ intern = False
1045
+ if intern:
1046
+ prefix = Naming.interned_prefixes['str']
1047
+ else:
1048
+ prefix = Naming.py_const_prefix
1049
+
1050
+ if encoding_key:
1051
+ encoding_prefix = '_%s' % encoding_key
1052
+ else:
1053
+ encoding_prefix = ''
1054
+
1055
+ pystring_cname = "%s%s%s_%s" % (
1056
+ prefix,
1057
+ (is_str and 's') or (is_unicode and 'u') or 'b',
1058
+ encoding_prefix,
1059
+ self.cname[len(Naming.const_prefix):])
1060
+
1061
+ py_string = PyStringConst(
1062
+ pystring_cname, encoding, is_unicode, is_str, py3str_cstring, intern)
1063
+ self.py_strings[key] = py_string
1064
+ return py_string
1065
+
1066
+ class PyStringConst(object):
1067
+ """Global info about a Python string constant held by GlobalState.
1068
+ """
1069
+ # cname string
1070
+ # py3str_cstring string
1071
+ # encoding string
1072
+ # intern boolean
1073
+ # is_unicode boolean
1074
+ # is_str boolean
1075
+
1076
+ def __init__(self, cname, encoding, is_unicode, is_str=False,
1077
+ py3str_cstring=None, intern=False):
1078
+ self.cname = cname
1079
+ self.py3str_cstring = py3str_cstring
1080
+ self.encoding = encoding
1081
+ self.is_str = is_str
1082
+ self.is_unicode = is_unicode
1083
+ self.intern = intern
1084
+
1085
+ def __lt__(self, other):
1086
+ return self.cname < other.cname
1087
+
1088
+
1089
+ class GlobalState(object):
1090
+ # filename_table {string : int} for finding filename table indexes
1091
+ # filename_list [string] filenames in filename table order
1092
+ # input_file_contents dict contents (=list of lines) of any file that was used as input
1093
+ # to create this output C code. This is
1094
+ # used to annotate the comments.
1095
+ #
1096
+ # utility_codes set IDs of used utility code (to avoid reinsertion)
1097
+ #
1098
+ # declared_cnames {string:Entry} used in a transition phase to merge pxd-declared
1099
+ # constants etc. into the pyx-declared ones (i.e,
1100
+ # check if constants are already added).
1101
+ # In time, hopefully the literals etc. will be
1102
+ # supplied directly instead.
1103
+ #
1104
+ # const_cnames_used dict global counter for unique constant identifiers
1105
+ #
1106
+
1107
+ # parts {string:CCodeWriter}
1108
+
1109
+
1110
+ # interned_strings
1111
+ # consts
1112
+ # interned_nums
1113
+
1114
+ # directives set Temporary variable used to track
1115
+ # the current set of directives in the code generation
1116
+ # process.
1117
+
1118
+ directives = {}
1119
+
1120
+ code_layout = [
1121
+ 'h_code',
1122
+ 'filename_table',
1123
+ 'utility_code_proto_before_types',
1124
+ 'numeric_typedefs', # Let these detailed individual parts stay!,
1125
+ 'complex_type_declarations', # as the proper solution is to make a full DAG...
1126
+ 'type_declarations', # More coarse-grained blocks would simply hide
1127
+ 'utility_code_proto', # the ugliness, not fix it
1128
+ 'module_declarations',
1129
+ 'typeinfo',
1130
+ 'before_global_var',
1131
+ 'global_var',
1132
+ 'string_decls',
1133
+ 'decls',
1134
+ 'late_includes',
1135
+ 'all_the_rest',
1136
+ 'pystring_table',
1137
+ 'cached_builtins',
1138
+ 'cached_constants',
1139
+ 'init_globals',
1140
+ 'init_module',
1141
+ 'cleanup_globals',
1142
+ 'cleanup_module',
1143
+ 'main_method',
1144
+ 'utility_code_def',
1145
+ 'end'
1146
+ ]
1147
+
1148
+
1149
+ def __init__(self, writer, module_node, code_config, common_utility_include_dir=None):
1150
+ self.filename_table = {}
1151
+ self.filename_list = []
1152
+ self.input_file_contents = {}
1153
+ self.utility_codes = set()
1154
+ self.declared_cnames = {}
1155
+ self.in_utility_code_generation = False
1156
+ self.code_config = code_config
1157
+ self.common_utility_include_dir = common_utility_include_dir
1158
+ self.parts = {}
1159
+ self.module_node = module_node # because some utility code generation needs it
1160
+ # (generating backwards-compatible Get/ReleaseBuffer
1161
+
1162
+ self.const_cnames_used = {}
1163
+ self.string_const_index = {}
1164
+ self.dedup_const_index = {}
1165
+ self.pyunicode_ptr_const_index = {}
1166
+ self.num_const_index = {}
1167
+ self.py_constants = []
1168
+ self.cached_cmethods = {}
1169
+ self.initialised_constants = set()
1170
+
1171
+ writer.set_global_state(self)
1172
+ self.rootwriter = writer
1173
+
1174
+ def initialize_main_c_code(self):
1175
+ rootwriter = self.rootwriter
1176
+ for part in self.code_layout:
1177
+ self.parts[part] = rootwriter.insertion_point()
1178
+
1179
+ if not Options.cache_builtins:
1180
+ del self.parts['cached_builtins']
1181
+ else:
1182
+ w = self.parts['cached_builtins']
1183
+ w.enter_cfunc_scope()
1184
+ w.putln("static CYTHON_SMALL_CODE int __Pyx_InitCachedBuiltins(void) {")
1185
+
1186
+ w = self.parts['cached_constants']
1187
+ w.enter_cfunc_scope()
1188
+ w.putln("")
1189
+ w.putln("static CYTHON_SMALL_CODE int __Pyx_InitCachedConstants(void) {")
1190
+ w.put_declare_refcount_context()
1191
+ w.put_setup_refcount_context("__Pyx_InitCachedConstants")
1192
+
1193
+ w = self.parts['init_globals']
1194
+ w.enter_cfunc_scope()
1195
+ w.putln("")
1196
+ w.putln("static CYTHON_SMALL_CODE int __Pyx_InitGlobals(void) {")
1197
+
1198
+ if not Options.generate_cleanup_code:
1199
+ del self.parts['cleanup_globals']
1200
+ else:
1201
+ w = self.parts['cleanup_globals']
1202
+ w.enter_cfunc_scope()
1203
+ w.putln("")
1204
+ w.putln("static CYTHON_SMALL_CODE void __Pyx_CleanupGlobals(void) {")
1205
+
1206
+ code = self.parts['utility_code_proto']
1207
+ code.putln("")
1208
+ code.putln("/* --- Runtime support code (head) --- */")
1209
+
1210
+ code = self.parts['utility_code_def']
1211
+ if self.code_config.emit_linenums:
1212
+ code.write('\n#line 1 "cython_utility"\n')
1213
+ code.putln("")
1214
+ code.putln("/* --- Runtime support code --- */")
1215
+
1216
+ def finalize_main_c_code(self):
1217
+ self.close_global_decls()
1218
+
1219
+ #
1220
+ # utility_code_def
1221
+ #
1222
+ code = self.parts['utility_code_def']
1223
+ util = TempitaUtilityCode.load_cached("TypeConversions", "TypeConversion.c")
1224
+ code.put(util.format_code(util.impl))
1225
+ code.putln("")
1226
+
1227
+ def __getitem__(self, key):
1228
+ return self.parts[key]
1229
+
1230
+ #
1231
+ # Global constants, interned objects, etc.
1232
+ #
1233
+ def close_global_decls(self):
1234
+ # This is called when it is known that no more global declarations will
1235
+ # declared.
1236
+ self.generate_const_declarations()
1237
+ if Options.cache_builtins:
1238
+ w = self.parts['cached_builtins']
1239
+ w.putln("return 0;")
1240
+ if w.label_used(w.error_label):
1241
+ w.put_label(w.error_label)
1242
+ w.putln("return -1;")
1243
+ w.putln("}")
1244
+ w.exit_cfunc_scope()
1245
+
1246
+ w = self.parts['cached_constants']
1247
+ w.put_finish_refcount_context()
1248
+ w.putln("return 0;")
1249
+ if w.label_used(w.error_label):
1250
+ w.put_label(w.error_label)
1251
+ w.put_finish_refcount_context()
1252
+ w.putln("return -1;")
1253
+ w.putln("}")
1254
+ w.exit_cfunc_scope()
1255
+
1256
+ w = self.parts['init_globals']
1257
+ w.putln("return 0;")
1258
+ if w.label_used(w.error_label):
1259
+ w.put_label(w.error_label)
1260
+ w.putln("return -1;")
1261
+ w.putln("}")
1262
+ w.exit_cfunc_scope()
1263
+
1264
+ if Options.generate_cleanup_code:
1265
+ w = self.parts['cleanup_globals']
1266
+ w.putln("}")
1267
+ w.exit_cfunc_scope()
1268
+
1269
+ if Options.generate_cleanup_code:
1270
+ w = self.parts['cleanup_module']
1271
+ w.putln("}")
1272
+ w.exit_cfunc_scope()
1273
+
1274
+ def put_pyobject_decl(self, entry):
1275
+ self['global_var'].putln("static PyObject *%s;" % entry.cname)
1276
+
1277
+ # constant handling at code generation time
1278
+
1279
+ def get_cached_constants_writer(self, target=None):
1280
+ if target is not None:
1281
+ if target in self.initialised_constants:
1282
+ # Return None on second/later calls to prevent duplicate creation code.
1283
+ return None
1284
+ self.initialised_constants.add(target)
1285
+ return self.parts['cached_constants']
1286
+
1287
+ def get_int_const(self, str_value, longness=False):
1288
+ py_type = longness and 'long' or 'int'
1289
+ try:
1290
+ c = self.num_const_index[(str_value, py_type)]
1291
+ except KeyError:
1292
+ c = self.new_num_const(str_value, py_type)
1293
+ return c
1294
+
1295
+ def get_float_const(self, str_value, value_code):
1296
+ try:
1297
+ c = self.num_const_index[(str_value, 'float')]
1298
+ except KeyError:
1299
+ c = self.new_num_const(str_value, 'float', value_code)
1300
+ return c
1301
+
1302
+ def get_py_const(self, type, prefix='', cleanup_level=None, dedup_key=None):
1303
+ if dedup_key is not None:
1304
+ const = self.dedup_const_index.get(dedup_key)
1305
+ if const is not None:
1306
+ return const
1307
+ # create a new Python object constant
1308
+ const = self.new_py_const(type, prefix)
1309
+ if cleanup_level is not None \
1310
+ and cleanup_level <= Options.generate_cleanup_code:
1311
+ cleanup_writer = self.parts['cleanup_globals']
1312
+ cleanup_writer.putln('Py_CLEAR(%s);' % const.cname)
1313
+ if dedup_key is not None:
1314
+ self.dedup_const_index[dedup_key] = const
1315
+ return const
1316
+
1317
+ def get_string_const(self, text, py_version=None):
1318
+ # return a C string constant, creating a new one if necessary
1319
+ if text.is_unicode:
1320
+ byte_string = text.utf8encode()
1321
+ else:
1322
+ byte_string = text.byteencode()
1323
+ try:
1324
+ c = self.string_const_index[byte_string]
1325
+ except KeyError:
1326
+ c = self.new_string_const(text, byte_string)
1327
+ c.add_py_version(py_version)
1328
+ return c
1329
+
1330
+ def get_pyunicode_ptr_const(self, text):
1331
+ # return a Py_UNICODE[] constant, creating a new one if necessary
1332
+ assert text.is_unicode
1333
+ try:
1334
+ c = self.pyunicode_ptr_const_index[text]
1335
+ except KeyError:
1336
+ c = self.pyunicode_ptr_const_index[text] = self.new_const_cname()
1337
+ return c
1338
+
1339
+ def get_py_string_const(self, text, identifier=None,
1340
+ is_str=False, unicode_value=None):
1341
+ # return a Python string constant, creating a new one if necessary
1342
+ py3str_cstring = None
1343
+ if is_str and unicode_value is not None \
1344
+ and unicode_value.utf8encode() != text.byteencode():
1345
+ py3str_cstring = self.get_string_const(unicode_value, py_version=3)
1346
+ c_string = self.get_string_const(text, py_version=2)
1347
+ else:
1348
+ c_string = self.get_string_const(text)
1349
+ py_string = c_string.get_py_string_const(
1350
+ text.encoding, identifier, is_str, py3str_cstring)
1351
+ return py_string
1352
+
1353
+ def get_interned_identifier(self, text):
1354
+ return self.get_py_string_const(text, identifier=True)
1355
+
1356
+ def new_string_const(self, text, byte_string):
1357
+ cname = self.new_string_const_cname(byte_string)
1358
+ c = StringConst(cname, text, byte_string)
1359
+ self.string_const_index[byte_string] = c
1360
+ return c
1361
+
1362
+ def new_num_const(self, value, py_type, value_code=None):
1363
+ cname = self.new_num_const_cname(value, py_type)
1364
+ c = NumConst(cname, value, py_type, value_code)
1365
+ self.num_const_index[(value, py_type)] = c
1366
+ return c
1367
+
1368
+ def new_py_const(self, type, prefix=''):
1369
+ cname = self.new_const_cname(prefix)
1370
+ c = PyObjectConst(cname, type)
1371
+ self.py_constants.append(c)
1372
+ return c
1373
+
1374
+ def new_string_const_cname(self, bytes_value):
1375
+ # Create a new globally-unique nice name for a C string constant.
1376
+ value = bytes_value.decode('ASCII', 'ignore')
1377
+ return self.new_const_cname(value=value)
1378
+
1379
+ def new_num_const_cname(self, value, py_type):
1380
+ if py_type == 'long':
1381
+ value += 'L'
1382
+ py_type = 'int'
1383
+ prefix = Naming.interned_prefixes[py_type]
1384
+ cname = "%s%s" % (prefix, value)
1385
+ cname = cname.replace('+', '_').replace('-', 'neg_').replace('.', '_')
1386
+ return cname
1387
+
1388
+ def new_const_cname(self, prefix='', value=''):
1389
+ value = replace_identifier('_', value)[:32].strip('_')
1390
+ used = self.const_cnames_used
1391
+ name_suffix = value
1392
+ while name_suffix in used:
1393
+ counter = used[value] = used[value] + 1
1394
+ name_suffix = '%s_%d' % (value, counter)
1395
+ used[name_suffix] = 1
1396
+ if prefix:
1397
+ prefix = Naming.interned_prefixes[prefix]
1398
+ else:
1399
+ prefix = Naming.const_prefix
1400
+ return "%s%s" % (prefix, name_suffix)
1401
+
1402
+ def get_cached_unbound_method(self, type_cname, method_name):
1403
+ key = (type_cname, method_name)
1404
+ try:
1405
+ cname = self.cached_cmethods[key]
1406
+ except KeyError:
1407
+ cname = self.cached_cmethods[key] = self.new_const_cname(
1408
+ 'umethod', '%s_%s' % (type_cname, method_name))
1409
+ return cname
1410
+
1411
+ def cached_unbound_method_call_code(self, obj_cname, type_cname, method_name, arg_cnames):
1412
+ # admittedly, not the best place to put this method, but it is reused by UtilityCode and ExprNodes ...
1413
+ utility_code_name = "CallUnboundCMethod%d" % len(arg_cnames)
1414
+ self.use_utility_code(UtilityCode.load_cached(utility_code_name, "ObjectHandling.c"))
1415
+ cache_cname = self.get_cached_unbound_method(type_cname, method_name)
1416
+ args = [obj_cname] + arg_cnames
1417
+ return "__Pyx_%s(&%s, %s)" % (
1418
+ utility_code_name,
1419
+ cache_cname,
1420
+ ', '.join(args),
1421
+ )
1422
+
1423
+ def add_cached_builtin_decl(self, entry):
1424
+ if entry.is_builtin and entry.is_const:
1425
+ if self.should_declare(entry.cname, entry):
1426
+ self.put_pyobject_decl(entry)
1427
+ w = self.parts['cached_builtins']
1428
+ condition = None
1429
+ if entry.name in non_portable_builtins_map:
1430
+ condition, replacement = non_portable_builtins_map[entry.name]
1431
+ w.putln('#if %s' % condition)
1432
+ self.put_cached_builtin_init(
1433
+ entry.pos, StringEncoding.EncodedString(replacement),
1434
+ entry.cname)
1435
+ w.putln('#else')
1436
+ self.put_cached_builtin_init(
1437
+ entry.pos, StringEncoding.EncodedString(entry.name),
1438
+ entry.cname)
1439
+ if condition:
1440
+ w.putln('#endif')
1441
+
1442
+ def put_cached_builtin_init(self, pos, name, cname):
1443
+ w = self.parts['cached_builtins']
1444
+ interned_cname = self.get_interned_identifier(name).cname
1445
+ self.use_utility_code(
1446
+ UtilityCode.load_cached("GetBuiltinName", "ObjectHandling.c"))
1447
+ w.putln('%s = __Pyx_GetBuiltinName(%s); if (!%s) %s' % (
1448
+ cname,
1449
+ interned_cname,
1450
+ cname,
1451
+ w.error_goto(pos)))
1452
+
1453
+ def generate_const_declarations(self):
1454
+ self.generate_cached_methods_decls()
1455
+ self.generate_string_constants()
1456
+ self.generate_num_constants()
1457
+ self.generate_object_constant_decls()
1458
+
1459
+ def generate_object_constant_decls(self):
1460
+ consts = [(len(c.cname), c.cname, c)
1461
+ for c in self.py_constants]
1462
+ consts.sort()
1463
+ decls_writer = self.parts['decls']
1464
+ for _, cname, c in consts:
1465
+ decls_writer.putln(
1466
+ "static %s;" % c.type.declaration_code(cname))
1467
+
1468
+ def generate_cached_methods_decls(self):
1469
+ if not self.cached_cmethods:
1470
+ return
1471
+
1472
+ decl = self.parts['decls']
1473
+ init = self.parts['init_globals']
1474
+ cnames = []
1475
+ for (type_cname, method_name), cname in sorted(self.cached_cmethods.items()):
1476
+ cnames.append(cname)
1477
+ method_name_cname = self.get_interned_identifier(StringEncoding.EncodedString(method_name)).cname
1478
+ decl.putln('static __Pyx_CachedCFunction %s = {0, &%s, 0, 0, 0};' % (
1479
+ cname, method_name_cname))
1480
+ # split type reference storage as it might not be static
1481
+ init.putln('%s.type = (PyObject*)&%s;' % (
1482
+ cname, type_cname))
1483
+
1484
+ if Options.generate_cleanup_code:
1485
+ cleanup = self.parts['cleanup_globals']
1486
+ for cname in cnames:
1487
+ cleanup.putln("Py_CLEAR(%s.method);" % cname)
1488
+
1489
+ def generate_string_constants(self):
1490
+ c_consts = [(len(c.cname), c.cname, c) for c in self.string_const_index.values()]
1491
+ c_consts.sort()
1492
+ py_strings = []
1493
+
1494
+ decls_writer = self.parts['string_decls']
1495
+ for _, cname, c in c_consts:
1496
+ conditional = False
1497
+ if c.py_versions and (2 not in c.py_versions or 3 not in c.py_versions):
1498
+ conditional = True
1499
+ decls_writer.putln("#if PY_MAJOR_VERSION %s 3" % (
1500
+ (2 in c.py_versions) and '<' or '>='))
1501
+ decls_writer.putln('static const char %s[] = "%s";' % (
1502
+ cname, StringEncoding.split_string_literal(c.escaped_value)))
1503
+ if conditional:
1504
+ decls_writer.putln("#endif")
1505
+ if c.py_strings is not None:
1506
+ for py_string in c.py_strings.values():
1507
+ py_strings.append((c.cname, len(py_string.cname), py_string))
1508
+
1509
+ for c, cname in sorted(self.pyunicode_ptr_const_index.items()):
1510
+ utf16_array, utf32_array = StringEncoding.encode_pyunicode_string(c)
1511
+ if utf16_array:
1512
+ # Narrow and wide representations differ
1513
+ decls_writer.putln("#ifdef Py_UNICODE_WIDE")
1514
+ decls_writer.putln("static Py_UNICODE %s[] = { %s };" % (cname, utf32_array))
1515
+ if utf16_array:
1516
+ decls_writer.putln("#else")
1517
+ decls_writer.putln("static Py_UNICODE %s[] = { %s };" % (cname, utf16_array))
1518
+ decls_writer.putln("#endif")
1519
+
1520
+ if py_strings:
1521
+ self.use_utility_code(UtilityCode.load_cached("InitStrings", "StringTools.c"))
1522
+ py_strings.sort()
1523
+ w = self.parts['pystring_table']
1524
+ w.putln("")
1525
+ w.putln("static __Pyx_StringTabEntry %s[] = {" % Naming.stringtab_cname)
1526
+ for c_cname, _, py_string in py_strings:
1527
+ if not py_string.is_str or not py_string.encoding or \
1528
+ py_string.encoding in ('ASCII', 'USASCII', 'US-ASCII',
1529
+ 'UTF8', 'UTF-8'):
1530
+ encoding = '0'
1531
+ else:
1532
+ encoding = '"%s"' % py_string.encoding.lower()
1533
+
1534
+ decls_writer.putln(
1535
+ "static PyObject *%s;" % py_string.cname)
1536
+ if py_string.py3str_cstring:
1537
+ w.putln("#if PY_MAJOR_VERSION >= 3")
1538
+ w.putln("{&%s, %s, sizeof(%s), %s, %d, %d, %d}," % (
1539
+ py_string.cname,
1540
+ py_string.py3str_cstring.cname,
1541
+ py_string.py3str_cstring.cname,
1542
+ '0', 1, 0,
1543
+ py_string.intern
1544
+ ))
1545
+ w.putln("#else")
1546
+ w.putln("{&%s, %s, sizeof(%s), %s, %d, %d, %d}," % (
1547
+ py_string.cname,
1548
+ c_cname,
1549
+ c_cname,
1550
+ encoding,
1551
+ py_string.is_unicode,
1552
+ py_string.is_str,
1553
+ py_string.intern
1554
+ ))
1555
+ if py_string.py3str_cstring:
1556
+ w.putln("#endif")
1557
+ w.putln("{0, 0, 0, 0, 0, 0, 0}")
1558
+ w.putln("};")
1559
+
1560
+ init_globals = self.parts['init_globals']
1561
+ init_globals.putln(
1562
+ "if (__Pyx_InitStrings(%s) < 0) %s;" % (
1563
+ Naming.stringtab_cname,
1564
+ init_globals.error_goto(self.module_pos)))
1565
+
1566
+ def generate_num_constants(self):
1567
+ consts = [(c.py_type, c.value[0] == '-', len(c.value), c.value, c.value_code, c)
1568
+ for c in self.num_const_index.values()]
1569
+ consts.sort()
1570
+ decls_writer = self.parts['decls']
1571
+ init_globals = self.parts['init_globals']
1572
+ for py_type, _, _, value, value_code, c in consts:
1573
+ cname = c.cname
1574
+ decls_writer.putln("static PyObject *%s;" % cname)
1575
+ if py_type == 'float':
1576
+ function = 'PyFloat_FromDouble(%s)'
1577
+ elif py_type == 'long':
1578
+ function = 'PyLong_FromString((char *)"%s", 0, 0)'
1579
+ elif Utils.long_literal(value):
1580
+ function = 'PyInt_FromString((char *)"%s", 0, 0)'
1581
+ elif len(value.lstrip('-')) > 4:
1582
+ function = "PyInt_FromLong(%sL)"
1583
+ else:
1584
+ function = "PyInt_FromLong(%s)"
1585
+ init_globals.putln('%s = %s; %s' % (
1586
+ cname, function % value_code,
1587
+ init_globals.error_goto_if_null(cname, self.module_pos)))
1588
+
1589
+ # The functions below are there in a transition phase only
1590
+ # and will be deprecated. They are called from Nodes.BlockNode.
1591
+ # The copy&paste duplication is intentional in order to be able
1592
+ # to see quickly how BlockNode worked, until this is replaced.
1593
+
1594
+ def should_declare(self, cname, entry):
1595
+ if cname in self.declared_cnames:
1596
+ other = self.declared_cnames[cname]
1597
+ assert str(entry.type) == str(other.type)
1598
+ assert entry.init == other.init
1599
+ return False
1600
+ else:
1601
+ self.declared_cnames[cname] = entry
1602
+ return True
1603
+
1604
+ #
1605
+ # File name state
1606
+ #
1607
+
1608
+ def lookup_filename(self, source_desc):
1609
+ entry = source_desc.get_filenametable_entry()
1610
+ try:
1611
+ index = self.filename_table[entry]
1612
+ except KeyError:
1613
+ index = len(self.filename_list)
1614
+ self.filename_list.append(source_desc)
1615
+ self.filename_table[entry] = index
1616
+ return index
1617
+
1618
+ def commented_file_contents(self, source_desc):
1619
+ try:
1620
+ return self.input_file_contents[source_desc]
1621
+ except KeyError:
1622
+ pass
1623
+ source_file = source_desc.get_lines(encoding='ASCII',
1624
+ error_handling='ignore')
1625
+ try:
1626
+ F = [u' * ' + line.rstrip().replace(
1627
+ u'*/', u'*[inserted by cython to avoid comment closer]/'
1628
+ ).replace(
1629
+ u'/*', u'/[inserted by cython to avoid comment start]*'
1630
+ )
1631
+ for line in source_file]
1632
+ finally:
1633
+ if hasattr(source_file, 'close'):
1634
+ source_file.close()
1635
+ if not F: F.append(u'')
1636
+ self.input_file_contents[source_desc] = F
1637
+ return F
1638
+
1639
+ #
1640
+ # Utility code state
1641
+ #
1642
+
1643
+ def use_utility_code(self, utility_code):
1644
+ """
1645
+ Adds code to the C file. utility_code should
1646
+ a) implement __eq__/__hash__ for the purpose of knowing whether the same
1647
+ code has already been included
1648
+ b) implement put_code, which takes a globalstate instance
1649
+
1650
+ See UtilityCode.
1651
+ """
1652
+ if utility_code and utility_code not in self.utility_codes:
1653
+ self.utility_codes.add(utility_code)
1654
+ utility_code.put_code(self)
1655
+
1656
+ def use_entry_utility_code(self, entry):
1657
+ if entry is None:
1658
+ return
1659
+ if entry.utility_code:
1660
+ self.use_utility_code(entry.utility_code)
1661
+ if entry.utility_code_definition:
1662
+ self.use_utility_code(entry.utility_code_definition)
1663
+
1664
+
1665
+ def funccontext_property(func):
1666
+ name = func.__name__
1667
+ attribute_of = operator.attrgetter(name)
1668
+ def get(self):
1669
+ return attribute_of(self.funcstate)
1670
+ def set(self, value):
1671
+ setattr(self.funcstate, name, value)
1672
+ return property(get, set)
1673
+
1674
+
1675
+ class CCodeConfig(object):
1676
+ # emit_linenums boolean write #line pragmas?
1677
+ # emit_code_comments boolean copy the original code into C comments?
1678
+ # c_line_in_traceback boolean append the c file and line number to the traceback for exceptions?
1679
+
1680
+ def __init__(self, emit_linenums=True, emit_code_comments=True, c_line_in_traceback=True):
1681
+ self.emit_code_comments = emit_code_comments
1682
+ self.emit_linenums = emit_linenums
1683
+ self.c_line_in_traceback = c_line_in_traceback
1684
+
1685
+
1686
+ class CCodeWriter(object):
1687
+ """
1688
+ Utility class to output C code.
1689
+
1690
+ When creating an insertion point one must care about the state that is
1691
+ kept:
1692
+ - formatting state (level, bol) is cloned and used in insertion points
1693
+ as well
1694
+ - labels, temps, exc_vars: One must construct a scope in which these can
1695
+ exist by calling enter_cfunc_scope/exit_cfunc_scope (these are for
1696
+ sanity checking and forward compatibility). Created insertion points
1697
+ looses this scope and cannot access it.
1698
+ - marker: Not copied to insertion point
1699
+ - filename_table, filename_list, input_file_contents: All codewriters
1700
+ coming from the same root share the same instances simultaneously.
1701
+ """
1702
+
1703
+ # f file output file
1704
+ # buffer StringIOTree
1705
+
1706
+ # level int indentation level
1707
+ # bol bool beginning of line?
1708
+ # marker string comment to emit before next line
1709
+ # funcstate FunctionState contains state local to a C function used for code
1710
+ # generation (labels and temps state etc.)
1711
+ # globalstate GlobalState contains state global for a C file (input file info,
1712
+ # utility code, declared constants etc.)
1713
+ # pyclass_stack list used during recursive code generation to pass information
1714
+ # about the current class one is in
1715
+ # code_config CCodeConfig configuration options for the C code writer
1716
+
1717
+ @cython.locals(create_from='CCodeWriter')
1718
+ def __init__(self, create_from=None, buffer=None, copy_formatting=False):
1719
+ if buffer is None: buffer = StringIOTree()
1720
+ self.buffer = buffer
1721
+ self.last_pos = None
1722
+ self.last_marked_pos = None
1723
+ self.pyclass_stack = []
1724
+
1725
+ self.funcstate = None
1726
+ self.globalstate = None
1727
+ self.code_config = None
1728
+ self.level = 0
1729
+ self.call_level = 0
1730
+ self.bol = 1
1731
+
1732
+ if create_from is not None:
1733
+ # Use same global state
1734
+ self.set_global_state(create_from.globalstate)
1735
+ self.funcstate = create_from.funcstate
1736
+ # Clone formatting state
1737
+ if copy_formatting:
1738
+ self.level = create_from.level
1739
+ self.bol = create_from.bol
1740
+ self.call_level = create_from.call_level
1741
+ self.last_pos = create_from.last_pos
1742
+ self.last_marked_pos = create_from.last_marked_pos
1743
+
1744
+ def create_new(self, create_from, buffer, copy_formatting):
1745
+ # polymorphic constructor -- very slightly more versatile
1746
+ # than using __class__
1747
+ result = CCodeWriter(create_from, buffer, copy_formatting)
1748
+ return result
1749
+
1750
+ def set_global_state(self, global_state):
1751
+ assert self.globalstate is None # prevent overwriting once it's set
1752
+ self.globalstate = global_state
1753
+ self.code_config = global_state.code_config
1754
+
1755
+ def copyto(self, f):
1756
+ self.buffer.copyto(f)
1757
+
1758
+ def getvalue(self):
1759
+ return self.buffer.getvalue()
1760
+
1761
+ def write(self, s):
1762
+ # also put invalid markers (lineno 0), to indicate that those lines
1763
+ # have no Cython source code correspondence
1764
+ cython_lineno = self.last_marked_pos[1] if self.last_marked_pos else 0
1765
+ self.buffer.markers.extend([cython_lineno] * s.count('\n'))
1766
+ self.buffer.write(s)
1767
+
1768
+ def insertion_point(self):
1769
+ other = self.create_new(create_from=self, buffer=self.buffer.insertion_point(), copy_formatting=True)
1770
+ return other
1771
+
1772
+ def new_writer(self):
1773
+ """
1774
+ Creates a new CCodeWriter connected to the same global state, which
1775
+ can later be inserted using insert.
1776
+ """
1777
+ return CCodeWriter(create_from=self)
1778
+
1779
+ def insert(self, writer):
1780
+ """
1781
+ Inserts the contents of another code writer (created with
1782
+ the same global state) in the current location.
1783
+
1784
+ It is ok to write to the inserted writer also after insertion.
1785
+ """
1786
+ assert writer.globalstate is self.globalstate
1787
+ self.buffer.insert(writer.buffer)
1788
+
1789
+ # Properties delegated to function scope
1790
+ @funccontext_property
1791
+ def label_counter(self): pass
1792
+ @funccontext_property
1793
+ def return_label(self): pass
1794
+ @funccontext_property
1795
+ def error_label(self): pass
1796
+ @funccontext_property
1797
+ def labels_used(self): pass
1798
+ @funccontext_property
1799
+ def continue_label(self): pass
1800
+ @funccontext_property
1801
+ def break_label(self): pass
1802
+ @funccontext_property
1803
+ def return_from_error_cleanup_label(self): pass
1804
+ @funccontext_property
1805
+ def yield_labels(self): pass
1806
+
1807
+ # Functions delegated to function scope
1808
+ def new_label(self, name=None): return self.funcstate.new_label(name)
1809
+ def new_error_label(self): return self.funcstate.new_error_label()
1810
+ def new_yield_label(self, *args): return self.funcstate.new_yield_label(*args)
1811
+ def get_loop_labels(self): return self.funcstate.get_loop_labels()
1812
+ def set_loop_labels(self, labels): return self.funcstate.set_loop_labels(labels)
1813
+ def new_loop_labels(self): return self.funcstate.new_loop_labels()
1814
+ def get_all_labels(self): return self.funcstate.get_all_labels()
1815
+ def set_all_labels(self, labels): return self.funcstate.set_all_labels(labels)
1816
+ def all_new_labels(self): return self.funcstate.all_new_labels()
1817
+ def use_label(self, lbl): return self.funcstate.use_label(lbl)
1818
+ def label_used(self, lbl): return self.funcstate.label_used(lbl)
1819
+
1820
+
1821
+ def enter_cfunc_scope(self, scope=None):
1822
+ self.funcstate = FunctionState(self, scope=scope)
1823
+
1824
+ def exit_cfunc_scope(self):
1825
+ self.funcstate = None
1826
+
1827
+ # constant handling
1828
+
1829
+ def get_py_int(self, str_value, longness):
1830
+ return self.globalstate.get_int_const(str_value, longness).cname
1831
+
1832
+ def get_py_float(self, str_value, value_code):
1833
+ return self.globalstate.get_float_const(str_value, value_code).cname
1834
+
1835
+ def get_py_const(self, type, prefix='', cleanup_level=None, dedup_key=None):
1836
+ return self.globalstate.get_py_const(type, prefix, cleanup_level, dedup_key).cname
1837
+
1838
+ def get_string_const(self, text):
1839
+ return self.globalstate.get_string_const(text).cname
1840
+
1841
+ def get_pyunicode_ptr_const(self, text):
1842
+ return self.globalstate.get_pyunicode_ptr_const(text)
1843
+
1844
+ def get_py_string_const(self, text, identifier=None,
1845
+ is_str=False, unicode_value=None):
1846
+ return self.globalstate.get_py_string_const(
1847
+ text, identifier, is_str, unicode_value).cname
1848
+
1849
+ def get_argument_default_const(self, type):
1850
+ return self.globalstate.get_py_const(type).cname
1851
+
1852
+ def intern(self, text):
1853
+ return self.get_py_string_const(text)
1854
+
1855
+ def intern_identifier(self, text):
1856
+ return self.get_py_string_const(text, identifier=True)
1857
+
1858
+ def get_cached_constants_writer(self, target=None):
1859
+ return self.globalstate.get_cached_constants_writer(target)
1860
+
1861
+ # code generation
1862
+
1863
+ def putln(self, code="", safe=False):
1864
+ if self.last_pos and self.bol:
1865
+ self.emit_marker()
1866
+ if self.code_config.emit_linenums and self.last_marked_pos:
1867
+ source_desc, line, _ = self.last_marked_pos
1868
+ self.write('\n#line %s "%s"\n' % (line, source_desc.get_escaped_description()))
1869
+ if code:
1870
+ if safe:
1871
+ self.put_safe(code)
1872
+ else:
1873
+ self.put(code)
1874
+ self.write("\n")
1875
+ self.bol = 1
1876
+
1877
+ def mark_pos(self, pos, trace=True):
1878
+ if pos is None:
1879
+ return
1880
+ if self.last_marked_pos and self.last_marked_pos[:2] == pos[:2]:
1881
+ return
1882
+ self.last_pos = (pos, trace)
1883
+
1884
+ def emit_marker(self):
1885
+ pos, trace = self.last_pos
1886
+ self.last_marked_pos = pos
1887
+ self.last_pos = None
1888
+ self.write("\n")
1889
+ if self.code_config.emit_code_comments:
1890
+ self.indent()
1891
+ self.write("/* %s */\n" % self._build_marker(pos))
1892
+ if trace and self.funcstate and self.funcstate.can_trace and self.globalstate.directives['linetrace']:
1893
+ self.indent()
1894
+ self.write('__Pyx_TraceLine(%d,%d,%s)\n' % (
1895
+ pos[1], not self.funcstate.gil_owned, self.error_goto(pos)))
1896
+
1897
+ def _build_marker(self, pos):
1898
+ source_desc, line, col = pos
1899
+ assert isinstance(source_desc, SourceDescriptor)
1900
+ contents = self.globalstate.commented_file_contents(source_desc)
1901
+ lines = contents[max(0, line-3):line] # line numbers start at 1
1902
+ lines[-1] += u' # <<<<<<<<<<<<<<'
1903
+ lines += contents[line:line+2]
1904
+ return u'"%s":%d\n%s\n' % (source_desc.get_escaped_description(), line, u'\n'.join(lines))
1905
+
1906
+ def put_safe(self, code):
1907
+ # put code, but ignore {}
1908
+ self.write(code)
1909
+ self.bol = 0
1910
+
1911
+ def put_or_include(self, code, name):
1912
+ include_dir = self.globalstate.common_utility_include_dir
1913
+ if include_dir and len(code) > 1024:
1914
+ include_file = "%s_%s.h" % (
1915
+ name, hashlib.md5(code.encode('utf8')).hexdigest())
1916
+ path = os.path.join(include_dir, include_file)
1917
+ if not os.path.exists(path):
1918
+ tmp_path = '%s.tmp%s' % (path, os.getpid())
1919
+ with closing(Utils.open_new_file(tmp_path)) as f:
1920
+ f.write(code)
1921
+ shutil.move(tmp_path, path)
1922
+ code = '#include "%s"\n' % path
1923
+ self.put(code)
1924
+
1925
+ def put(self, code):
1926
+ fix_indent = False
1927
+ if "{" in code:
1928
+ dl = code.count("{")
1929
+ else:
1930
+ dl = 0
1931
+ if "}" in code:
1932
+ dl -= code.count("}")
1933
+ if dl < 0:
1934
+ self.level += dl
1935
+ elif dl == 0 and code[0] == "}":
1936
+ # special cases like "} else {" need a temporary dedent
1937
+ fix_indent = True
1938
+ self.level -= 1
1939
+ if self.bol:
1940
+ self.indent()
1941
+ self.write(code)
1942
+ self.bol = 0
1943
+ if dl > 0:
1944
+ self.level += dl
1945
+ elif fix_indent:
1946
+ self.level += 1
1947
+
1948
+ def putln_tempita(self, code, **context):
1949
+ from ..Tempita import sub
1950
+ self.putln(sub(code, **context))
1951
+
1952
+ def put_tempita(self, code, **context):
1953
+ from ..Tempita import sub
1954
+ self.put(sub(code, **context))
1955
+
1956
+ def increase_indent(self):
1957
+ self.level += 1
1958
+
1959
+ def decrease_indent(self):
1960
+ self.level -= 1
1961
+
1962
+ def begin_block(self):
1963
+ self.putln("{")
1964
+ self.increase_indent()
1965
+
1966
+ def end_block(self):
1967
+ self.decrease_indent()
1968
+ self.putln("}")
1969
+
1970
+ def indent(self):
1971
+ self.write(" " * self.level)
1972
+
1973
+ def get_py_version_hex(self, pyversion):
1974
+ return "0x%02X%02X%02X%02X" % (tuple(pyversion) + (0,0,0,0))[:4]
1975
+
1976
+ def put_label(self, lbl):
1977
+ if lbl in self.funcstate.labels_used:
1978
+ self.putln("%s:;" % lbl)
1979
+
1980
+ def put_goto(self, lbl):
1981
+ self.funcstate.use_label(lbl)
1982
+ self.putln("goto %s;" % lbl)
1983
+
1984
+ def put_var_declaration(self, entry, storage_class="",
1985
+ dll_linkage=None, definition=True):
1986
+ #print "Code.put_var_declaration:", entry.name, "definition =", definition ###
1987
+ if entry.visibility == 'private' and not (definition or entry.defined_in_pxd):
1988
+ #print "...private and not definition, skipping", entry.cname ###
1989
+ return
1990
+ if entry.visibility == "private" and not entry.used:
1991
+ #print "...private and not used, skipping", entry.cname ###
1992
+ return
1993
+ if storage_class:
1994
+ self.put("%s " % storage_class)
1995
+ if not entry.cf_used:
1996
+ self.put('CYTHON_UNUSED ')
1997
+ self.put(entry.type.declaration_code(
1998
+ entry.cname, dll_linkage=dll_linkage))
1999
+ if entry.init is not None:
2000
+ self.put_safe(" = %s" % entry.type.literal_code(entry.init))
2001
+ elif entry.type.is_pyobject:
2002
+ self.put(" = NULL")
2003
+ self.putln(";")
2004
+
2005
+ def put_temp_declarations(self, func_context):
2006
+ for name, type, manage_ref, static in func_context.temps_allocated:
2007
+ decl = type.declaration_code(name)
2008
+ if type.is_pyobject:
2009
+ self.putln("%s = NULL;" % decl)
2010
+ elif type.is_memoryviewslice:
2011
+ from . import MemoryView
2012
+ self.putln("%s = %s;" % (decl, MemoryView.memslice_entry_init))
2013
+ else:
2014
+ self.putln("%s%s;" % (static and "static " or "", decl))
2015
+
2016
+ if func_context.should_declare_error_indicator:
2017
+ if self.funcstate.uses_error_indicator:
2018
+ unused = ''
2019
+ else:
2020
+ unused = 'CYTHON_UNUSED '
2021
+ # Initialize these variables to silence compiler warnings
2022
+ self.putln("%sint %s = 0;" % (unused, Naming.lineno_cname))
2023
+ self.putln("%sconst char *%s = NULL;" % (unused, Naming.filename_cname))
2024
+ self.putln("%sint %s = 0;" % (unused, Naming.clineno_cname))
2025
+
2026
+ def put_generated_by(self):
2027
+ self.putln("/* Generated by Cython %s */" % Version.watermark)
2028
+ self.putln("")
2029
+
2030
+ def put_h_guard(self, guard):
2031
+ self.putln("#ifndef %s" % guard)
2032
+ self.putln("#define %s" % guard)
2033
+
2034
+ def unlikely(self, cond):
2035
+ if Options.gcc_branch_hints:
2036
+ return 'unlikely(%s)' % cond
2037
+ else:
2038
+ return cond
2039
+
2040
+ def build_function_modifiers(self, modifiers, mapper=modifier_output_mapper):
2041
+ if not modifiers:
2042
+ return ''
2043
+ return '%s ' % ' '.join([mapper(m,m) for m in modifiers])
2044
+
2045
+ # Python objects and reference counting
2046
+
2047
+ def entry_as_pyobject(self, entry):
2048
+ type = entry.type
2049
+ if (not entry.is_self_arg and not entry.type.is_complete()
2050
+ or entry.type.is_extension_type):
2051
+ return "(PyObject *)" + entry.cname
2052
+ else:
2053
+ return entry.cname
2054
+
2055
+ def as_pyobject(self, cname, type):
2056
+ from .PyrexTypes import py_object_type, typecast
2057
+ return typecast(py_object_type, type, cname)
2058
+
2059
+ def put_gotref(self, cname):
2060
+ self.putln("__Pyx_GOTREF(%s);" % cname)
2061
+
2062
+ def put_giveref(self, cname):
2063
+ self.putln("__Pyx_GIVEREF(%s);" % cname)
2064
+
2065
+ def put_xgiveref(self, cname):
2066
+ self.putln("__Pyx_XGIVEREF(%s);" % cname)
2067
+
2068
+ def put_xgotref(self, cname):
2069
+ self.putln("__Pyx_XGOTREF(%s);" % cname)
2070
+
2071
+ def put_incref(self, cname, type, nanny=True):
2072
+ if nanny:
2073
+ self.putln("__Pyx_INCREF(%s);" % self.as_pyobject(cname, type))
2074
+ else:
2075
+ self.putln("Py_INCREF(%s);" % self.as_pyobject(cname, type))
2076
+
2077
+ def put_decref(self, cname, type, nanny=True):
2078
+ self._put_decref(cname, type, nanny, null_check=False, clear=False)
2079
+
2080
+ def put_var_gotref(self, entry):
2081
+ if entry.type.is_pyobject:
2082
+ self.putln("__Pyx_GOTREF(%s);" % self.entry_as_pyobject(entry))
2083
+
2084
+ def put_var_giveref(self, entry):
2085
+ if entry.type.is_pyobject:
2086
+ self.putln("__Pyx_GIVEREF(%s);" % self.entry_as_pyobject(entry))
2087
+
2088
+ def put_var_xgotref(self, entry):
2089
+ if entry.type.is_pyobject:
2090
+ self.putln("__Pyx_XGOTREF(%s);" % self.entry_as_pyobject(entry))
2091
+
2092
+ def put_var_xgiveref(self, entry):
2093
+ if entry.type.is_pyobject:
2094
+ self.putln("__Pyx_XGIVEREF(%s);" % self.entry_as_pyobject(entry))
2095
+
2096
+ def put_var_incref(self, entry, nanny=True):
2097
+ if entry.type.is_pyobject:
2098
+ if nanny:
2099
+ self.putln("__Pyx_INCREF(%s);" % self.entry_as_pyobject(entry))
2100
+ else:
2101
+ self.putln("Py_INCREF(%s);" % self.entry_as_pyobject(entry))
2102
+
2103
+ def put_var_xincref(self, entry):
2104
+ if entry.type.is_pyobject:
2105
+ self.putln("__Pyx_XINCREF(%s);" % self.entry_as_pyobject(entry))
2106
+
2107
+ def put_decref_clear(self, cname, type, nanny=True, clear_before_decref=False):
2108
+ self._put_decref(cname, type, nanny, null_check=False,
2109
+ clear=True, clear_before_decref=clear_before_decref)
2110
+
2111
+ def put_xdecref(self, cname, type, nanny=True, have_gil=True):
2112
+ self._put_decref(cname, type, nanny, null_check=True,
2113
+ have_gil=have_gil, clear=False)
2114
+
2115
+ def put_xdecref_clear(self, cname, type, nanny=True, clear_before_decref=False):
2116
+ self._put_decref(cname, type, nanny, null_check=True,
2117
+ clear=True, clear_before_decref=clear_before_decref)
2118
+
2119
+ def _put_decref(self, cname, type, nanny=True, null_check=False,
2120
+ have_gil=True, clear=False, clear_before_decref=False):
2121
+ if type.is_memoryviewslice:
2122
+ self.put_xdecref_memoryviewslice(cname, have_gil=have_gil)
2123
+ return
2124
+
2125
+ prefix = '__Pyx' if nanny else 'Py'
2126
+ X = 'X' if null_check else ''
2127
+
2128
+ if clear:
2129
+ if clear_before_decref:
2130
+ if not nanny:
2131
+ X = '' # CPython doesn't have a Py_XCLEAR()
2132
+ self.putln("%s_%sCLEAR(%s);" % (prefix, X, cname))
2133
+ else:
2134
+ self.putln("%s_%sDECREF(%s); %s = 0;" % (
2135
+ prefix, X, self.as_pyobject(cname, type), cname))
2136
+ else:
2137
+ self.putln("%s_%sDECREF(%s);" % (
2138
+ prefix, X, self.as_pyobject(cname, type)))
2139
+
2140
+ def put_decref_set(self, cname, rhs_cname):
2141
+ self.putln("__Pyx_DECREF_SET(%s, %s);" % (cname, rhs_cname))
2142
+
2143
+ def put_xdecref_set(self, cname, rhs_cname):
2144
+ self.putln("__Pyx_XDECREF_SET(%s, %s);" % (cname, rhs_cname))
2145
+
2146
+ def put_var_decref(self, entry):
2147
+ if entry.type.is_pyobject:
2148
+ self.putln("__Pyx_XDECREF(%s);" % self.entry_as_pyobject(entry))
2149
+
2150
+ def put_var_xdecref(self, entry, nanny=True):
2151
+ if entry.type.is_pyobject:
2152
+ if nanny:
2153
+ self.putln("__Pyx_XDECREF(%s);" % self.entry_as_pyobject(entry))
2154
+ else:
2155
+ self.putln("Py_XDECREF(%s);" % self.entry_as_pyobject(entry))
2156
+
2157
+ def put_var_decref_clear(self, entry):
2158
+ self._put_var_decref_clear(entry, null_check=False)
2159
+
2160
+ def put_var_xdecref_clear(self, entry):
2161
+ self._put_var_decref_clear(entry, null_check=True)
2162
+
2163
+ def _put_var_decref_clear(self, entry, null_check):
2164
+ if entry.type.is_pyobject:
2165
+ if entry.in_closure:
2166
+ # reset before DECREF to make sure closure state is
2167
+ # consistent during call to DECREF()
2168
+ self.putln("__Pyx_%sCLEAR(%s);" % (
2169
+ null_check and 'X' or '',
2170
+ entry.cname))
2171
+ else:
2172
+ self.putln("__Pyx_%sDECREF(%s); %s = 0;" % (
2173
+ null_check and 'X' or '',
2174
+ self.entry_as_pyobject(entry),
2175
+ entry.cname))
2176
+
2177
+ def put_var_decrefs(self, entries, used_only = 0):
2178
+ for entry in entries:
2179
+ if not used_only or entry.used:
2180
+ if entry.xdecref_cleanup:
2181
+ self.put_var_xdecref(entry)
2182
+ else:
2183
+ self.put_var_decref(entry)
2184
+
2185
+ def put_var_xdecrefs(self, entries):
2186
+ for entry in entries:
2187
+ self.put_var_xdecref(entry)
2188
+
2189
+ def put_var_xdecrefs_clear(self, entries):
2190
+ for entry in entries:
2191
+ self.put_var_xdecref_clear(entry)
2192
+
2193
+ def put_incref_memoryviewslice(self, slice_cname, have_gil=False):
2194
+ from . import MemoryView
2195
+ self.globalstate.use_utility_code(MemoryView.memviewslice_init_code)
2196
+ self.putln("__PYX_INC_MEMVIEW(&%s, %d);" % (slice_cname, int(have_gil)))
2197
+
2198
+ def put_xdecref_memoryviewslice(self, slice_cname, have_gil=False):
2199
+ from . import MemoryView
2200
+ self.globalstate.use_utility_code(MemoryView.memviewslice_init_code)
2201
+ self.putln("__PYX_XDEC_MEMVIEW(&%s, %d);" % (slice_cname, int(have_gil)))
2202
+
2203
+ def put_xgiveref_memoryviewslice(self, slice_cname):
2204
+ self.put_xgiveref("%s.memview" % slice_cname)
2205
+
2206
+ def put_init_to_py_none(self, cname, type, nanny=True):
2207
+ from .PyrexTypes import py_object_type, typecast
2208
+ py_none = typecast(type, py_object_type, "Py_None")
2209
+ if nanny:
2210
+ self.putln("%s = %s; __Pyx_INCREF(Py_None);" % (cname, py_none))
2211
+ else:
2212
+ self.putln("%s = %s; Py_INCREF(Py_None);" % (cname, py_none))
2213
+
2214
+ def put_init_var_to_py_none(self, entry, template = "%s", nanny=True):
2215
+ code = template % entry.cname
2216
+ #if entry.type.is_extension_type:
2217
+ # code = "((PyObject*)%s)" % code
2218
+ self.put_init_to_py_none(code, entry.type, nanny)
2219
+ if entry.in_closure:
2220
+ self.put_giveref('Py_None')
2221
+
2222
+ def put_pymethoddef(self, entry, term, allow_skip=True, wrapper_code_writer=None):
2223
+ if entry.is_special or entry.name == '__getattribute__':
2224
+ if entry.name not in special_py_methods:
2225
+ if entry.name == '__getattr__' and not self.globalstate.directives['fast_getattr']:
2226
+ pass
2227
+ # Python's typeobject.c will automatically fill in our slot
2228
+ # in add_operators() (called by PyType_Ready) with a value
2229
+ # that's better than ours.
2230
+ elif allow_skip:
2231
+ return
2232
+
2233
+ method_flags = entry.signature.method_flags()
2234
+ if not method_flags:
2235
+ return
2236
+ if entry.is_special:
2237
+ from . import TypeSlots
2238
+ method_flags += [TypeSlots.method_coexist]
2239
+ func_ptr = wrapper_code_writer.put_pymethoddef_wrapper(entry) if wrapper_code_writer else entry.func_cname
2240
+ # Add required casts, but try not to shadow real warnings.
2241
+ cast = '__Pyx_PyCFunctionFast' if 'METH_FASTCALL' in method_flags else 'PyCFunction'
2242
+ if 'METH_KEYWORDS' in method_flags:
2243
+ cast += 'WithKeywords'
2244
+ if cast != 'PyCFunction':
2245
+ func_ptr = '(void*)(%s)%s' % (cast, func_ptr)
2246
+ self.putln(
2247
+ '{"%s", (PyCFunction)%s, %s, %s}%s' % (
2248
+ entry.name,
2249
+ func_ptr,
2250
+ "|".join(method_flags),
2251
+ entry.doc_cname if entry.doc else '0',
2252
+ term))
2253
+
2254
+ def put_pymethoddef_wrapper(self, entry):
2255
+ func_cname = entry.func_cname
2256
+ if entry.is_special:
2257
+ method_flags = entry.signature.method_flags()
2258
+ if method_flags and 'METH_NOARGS' in method_flags:
2259
+ # Special NOARGS methods really take no arguments besides 'self', but PyCFunction expects one.
2260
+ func_cname = Naming.method_wrapper_prefix + func_cname
2261
+ self.putln("static PyObject *%s(PyObject *self, CYTHON_UNUSED PyObject *arg) {return %s(self);}" % (
2262
+ func_cname, entry.func_cname))
2263
+ return func_cname
2264
+
2265
+ # GIL methods
2266
+
2267
+ def put_ensure_gil(self, declare_gilstate=True, variable=None):
2268
+ """
2269
+ Acquire the GIL. The generated code is safe even when no PyThreadState
2270
+ has been allocated for this thread (for threads not initialized by
2271
+ using the Python API). Additionally, the code generated by this method
2272
+ may be called recursively.
2273
+ """
2274
+ self.globalstate.use_utility_code(
2275
+ UtilityCode.load_cached("ForceInitThreads", "ModuleSetupCode.c"))
2276
+ if self.globalstate.directives['fast_gil']:
2277
+ self.globalstate.use_utility_code(UtilityCode.load_cached("FastGil", "ModuleSetupCode.c"))
2278
+ else:
2279
+ self.globalstate.use_utility_code(UtilityCode.load_cached("NoFastGil", "ModuleSetupCode.c"))
2280
+ self.putln("#ifdef WITH_THREAD")
2281
+ if not variable:
2282
+ variable = '__pyx_gilstate_save'
2283
+ if declare_gilstate:
2284
+ self.put("PyGILState_STATE ")
2285
+ self.putln("%s = __Pyx_PyGILState_Ensure();" % variable)
2286
+ self.putln("#endif")
2287
+
2288
+ def put_release_ensured_gil(self, variable=None):
2289
+ """
2290
+ Releases the GIL, corresponds to `put_ensure_gil`.
2291
+ """
2292
+ if self.globalstate.directives['fast_gil']:
2293
+ self.globalstate.use_utility_code(UtilityCode.load_cached("FastGil", "ModuleSetupCode.c"))
2294
+ else:
2295
+ self.globalstate.use_utility_code(UtilityCode.load_cached("NoFastGil", "ModuleSetupCode.c"))
2296
+ if not variable:
2297
+ variable = '__pyx_gilstate_save'
2298
+ self.putln("#ifdef WITH_THREAD")
2299
+ self.putln("__Pyx_PyGILState_Release(%s);" % variable)
2300
+ self.putln("#endif")
2301
+
2302
+ def put_acquire_gil(self, variable=None):
2303
+ """
2304
+ Acquire the GIL. The thread's thread state must have been initialized
2305
+ by a previous `put_release_gil`
2306
+ """
2307
+ if self.globalstate.directives['fast_gil']:
2308
+ self.globalstate.use_utility_code(UtilityCode.load_cached("FastGil", "ModuleSetupCode.c"))
2309
+ else:
2310
+ self.globalstate.use_utility_code(UtilityCode.load_cached("NoFastGil", "ModuleSetupCode.c"))
2311
+ self.putln("#ifdef WITH_THREAD")
2312
+ self.putln("__Pyx_FastGIL_Forget();")
2313
+ if variable:
2314
+ self.putln('_save = %s;' % variable)
2315
+ self.putln("Py_BLOCK_THREADS")
2316
+ self.putln("#endif")
2317
+
2318
+ def put_release_gil(self, variable=None):
2319
+ "Release the GIL, corresponds to `put_acquire_gil`."
2320
+ if self.globalstate.directives['fast_gil']:
2321
+ self.globalstate.use_utility_code(UtilityCode.load_cached("FastGil", "ModuleSetupCode.c"))
2322
+ else:
2323
+ self.globalstate.use_utility_code(UtilityCode.load_cached("NoFastGil", "ModuleSetupCode.c"))
2324
+ self.putln("#ifdef WITH_THREAD")
2325
+ self.putln("PyThreadState *_save;")
2326
+ self.putln("Py_UNBLOCK_THREADS")
2327
+ if variable:
2328
+ self.putln('%s = _save;' % variable)
2329
+ self.putln("__Pyx_FastGIL_Remember();")
2330
+ self.putln("#endif")
2331
+
2332
+ def declare_gilstate(self):
2333
+ self.putln("#ifdef WITH_THREAD")
2334
+ self.putln("PyGILState_STATE __pyx_gilstate_save;")
2335
+ self.putln("#endif")
2336
+
2337
+ # error handling
2338
+
2339
+ def put_error_if_neg(self, pos, value):
2340
+ # TODO this path is almost _never_ taken, yet this macro makes is slower!
2341
+ # return self.putln("if (unlikely(%s < 0)) %s" % (value, self.error_goto(pos)))
2342
+ return self.putln("if (%s < 0) %s" % (value, self.error_goto(pos)))
2343
+
2344
+ def put_error_if_unbound(self, pos, entry, in_nogil_context=False):
2345
+ from . import ExprNodes
2346
+ if entry.from_closure:
2347
+ func = '__Pyx_RaiseClosureNameError'
2348
+ self.globalstate.use_utility_code(
2349
+ ExprNodes.raise_closure_name_error_utility_code)
2350
+ elif entry.type.is_memoryviewslice and in_nogil_context:
2351
+ func = '__Pyx_RaiseUnboundMemoryviewSliceNogil'
2352
+ self.globalstate.use_utility_code(
2353
+ ExprNodes.raise_unbound_memoryview_utility_code_nogil)
2354
+ else:
2355
+ func = '__Pyx_RaiseUnboundLocalError'
2356
+ self.globalstate.use_utility_code(
2357
+ ExprNodes.raise_unbound_local_error_utility_code)
2358
+
2359
+ self.putln('if (unlikely(!%s)) { %s("%s"); %s }' % (
2360
+ entry.type.check_for_null_code(entry.cname),
2361
+ func,
2362
+ entry.name,
2363
+ self.error_goto(pos)))
2364
+
2365
+ def set_error_info(self, pos, used=False):
2366
+ self.funcstate.should_declare_error_indicator = True
2367
+ if used:
2368
+ self.funcstate.uses_error_indicator = True
2369
+ return "__PYX_MARK_ERR_POS(%s, %s)" % (
2370
+ self.lookup_filename(pos[0]),
2371
+ pos[1])
2372
+
2373
+ def error_goto(self, pos, used=True):
2374
+ lbl = self.funcstate.error_label
2375
+ self.funcstate.use_label(lbl)
2376
+ if pos is None:
2377
+ return 'goto %s;' % lbl
2378
+ self.funcstate.should_declare_error_indicator = True
2379
+ if used:
2380
+ self.funcstate.uses_error_indicator = True
2381
+ return "__PYX_ERR(%s, %s, %s)" % (
2382
+ self.lookup_filename(pos[0]),
2383
+ pos[1],
2384
+ lbl)
2385
+
2386
+ def error_goto_if(self, cond, pos):
2387
+ return "if (%s) %s" % (self.unlikely(cond), self.error_goto(pos))
2388
+
2389
+ def error_goto_if_null(self, cname, pos):
2390
+ return self.error_goto_if("!%s" % cname, pos)
2391
+
2392
+ def error_goto_if_neg(self, cname, pos):
2393
+ return self.error_goto_if("%s < 0" % cname, pos)
2394
+
2395
+ def error_goto_if_PyErr(self, pos):
2396
+ return self.error_goto_if("PyErr_Occurred()", pos)
2397
+
2398
+ def lookup_filename(self, filename):
2399
+ return self.globalstate.lookup_filename(filename)
2400
+
2401
+ def put_declare_refcount_context(self):
2402
+ self.putln('__Pyx_RefNannyDeclarations')
2403
+
2404
+ def put_setup_refcount_context(self, name, acquire_gil=False):
2405
+ if acquire_gil:
2406
+ self.globalstate.use_utility_code(
2407
+ UtilityCode.load_cached("ForceInitThreads", "ModuleSetupCode.c"))
2408
+ self.putln('__Pyx_RefNannySetupContext("%s", %d);' % (name, acquire_gil and 1 or 0))
2409
+
2410
+ def put_finish_refcount_context(self):
2411
+ self.putln("__Pyx_RefNannyFinishContext();")
2412
+
2413
+ def put_add_traceback(self, qualified_name, include_cline=True):
2414
+ """
2415
+ Build a Python traceback for propagating exceptions.
2416
+
2417
+ qualified_name should be the qualified name of the function.
2418
+ """
2419
+ format_tuple = (
2420
+ qualified_name,
2421
+ Naming.clineno_cname if include_cline else 0,
2422
+ Naming.lineno_cname,
2423
+ Naming.filename_cname,
2424
+ )
2425
+ self.funcstate.uses_error_indicator = True
2426
+ self.putln('__Pyx_AddTraceback("%s", %s, %s, %s);' % format_tuple)
2427
+
2428
+ def put_unraisable(self, qualified_name, nogil=False):
2429
+ """
2430
+ Generate code to print a Python warning for an unraisable exception.
2431
+
2432
+ qualified_name should be the qualified name of the function.
2433
+ """
2434
+ format_tuple = (
2435
+ qualified_name,
2436
+ Naming.clineno_cname,
2437
+ Naming.lineno_cname,
2438
+ Naming.filename_cname,
2439
+ self.globalstate.directives['unraisable_tracebacks'],
2440
+ nogil,
2441
+ )
2442
+ self.funcstate.uses_error_indicator = True
2443
+ self.putln('__Pyx_WriteUnraisable("%s", %s, %s, %s, %d, %d);' % format_tuple)
2444
+ self.globalstate.use_utility_code(
2445
+ UtilityCode.load_cached("WriteUnraisableException", "Exceptions.c"))
2446
+
2447
+ def put_trace_declarations(self):
2448
+ self.putln('__Pyx_TraceDeclarations')
2449
+
2450
+ def put_trace_frame_init(self, codeobj=None):
2451
+ if codeobj:
2452
+ self.putln('__Pyx_TraceFrameInit(%s)' % codeobj)
2453
+
2454
+ def put_trace_call(self, name, pos, nogil=False):
2455
+ self.putln('__Pyx_TraceCall("%s", %s[%s], %s, %d, %s);' % (
2456
+ name, Naming.filetable_cname, self.lookup_filename(pos[0]), pos[1], nogil, self.error_goto(pos)))
2457
+
2458
+ def put_trace_exception(self):
2459
+ self.putln("__Pyx_TraceException();")
2460
+
2461
+ def put_trace_return(self, retvalue_cname, nogil=False):
2462
+ self.putln("__Pyx_TraceReturn(%s, %d);" % (retvalue_cname, nogil))
2463
+
2464
+ def putln_openmp(self, string):
2465
+ self.putln("#ifdef _OPENMP")
2466
+ self.putln(string)
2467
+ self.putln("#endif /* _OPENMP */")
2468
+
2469
+ def undef_builtin_expect(self, cond):
2470
+ """
2471
+ Redefine the macros likely() and unlikely to no-ops, depending on
2472
+ condition 'cond'
2473
+ """
2474
+ self.putln("#if %s" % cond)
2475
+ self.putln(" #undef likely")
2476
+ self.putln(" #undef unlikely")
2477
+ self.putln(" #define likely(x) (x)")
2478
+ self.putln(" #define unlikely(x) (x)")
2479
+ self.putln("#endif")
2480
+
2481
+ def redef_builtin_expect(self, cond):
2482
+ self.putln("#if %s" % cond)
2483
+ self.putln(" #undef likely")
2484
+ self.putln(" #undef unlikely")
2485
+ self.putln(" #define likely(x) __builtin_expect(!!(x), 1)")
2486
+ self.putln(" #define unlikely(x) __builtin_expect(!!(x), 0)")
2487
+ self.putln("#endif")
2488
+
2489
+
2490
+ class PyrexCodeWriter(object):
2491
+ # f file output file
2492
+ # level int indentation level
2493
+
2494
+ def __init__(self, outfile_name):
2495
+ self.f = Utils.open_new_file(outfile_name)
2496
+ self.level = 0
2497
+
2498
+ def putln(self, code):
2499
+ self.f.write("%s%s\n" % (" " * self.level, code))
2500
+
2501
+ def indent(self):
2502
+ self.level += 1
2503
+
2504
+ def dedent(self):
2505
+ self.level -= 1
2506
+
2507
+ class PyxCodeWriter(object):
2508
+ """
2509
+ Can be used for writing out some Cython code. To use the indenter
2510
+ functionality, the Cython.Compiler.Importer module will have to be used
2511
+ to load the code to support python 2.4
2512
+ """
2513
+
2514
+ def __init__(self, buffer=None, indent_level=0, context=None, encoding='ascii'):
2515
+ self.buffer = buffer or StringIOTree()
2516
+ self.level = indent_level
2517
+ self.context = context
2518
+ self.encoding = encoding
2519
+
2520
+ def indent(self, levels=1):
2521
+ self.level += levels
2522
+ return True
2523
+
2524
+ def dedent(self, levels=1):
2525
+ self.level -= levels
2526
+
2527
+ def indenter(self, line):
2528
+ """
2529
+ Instead of
2530
+
2531
+ with pyx_code.indenter("for i in range(10):"):
2532
+ pyx_code.putln("print i")
2533
+
2534
+ write
2535
+
2536
+ if pyx_code.indenter("for i in range(10);"):
2537
+ pyx_code.putln("print i")
2538
+ pyx_code.dedent()
2539
+ """
2540
+ self.putln(line)
2541
+ self.indent()
2542
+ return True
2543
+
2544
+ def getvalue(self):
2545
+ result = self.buffer.getvalue()
2546
+ if isinstance(result, bytes):
2547
+ result = result.decode(self.encoding)
2548
+ return result
2549
+
2550
+ def putln(self, line, context=None):
2551
+ context = context or self.context
2552
+ if context:
2553
+ line = sub_tempita(line, context)
2554
+ self._putln(line)
2555
+
2556
+ def _putln(self, line):
2557
+ self.buffer.write("%s%s\n" % (self.level * " ", line))
2558
+
2559
+ def put_chunk(self, chunk, context=None):
2560
+ context = context or self.context
2561
+ if context:
2562
+ chunk = sub_tempita(chunk, context)
2563
+
2564
+ chunk = textwrap.dedent(chunk)
2565
+ for line in chunk.splitlines():
2566
+ self._putln(line)
2567
+
2568
+ def insertion_point(self):
2569
+ return PyxCodeWriter(self.buffer.insertion_point(), self.level,
2570
+ self.context)
2571
+
2572
+ def named_insertion_point(self, name):
2573
+ setattr(self, name, self.insertion_point())
2574
+
2575
+
2576
+ class ClosureTempAllocator(object):
2577
+ def __init__(self, klass):
2578
+ self.klass = klass
2579
+ self.temps_allocated = {}
2580
+ self.temps_free = {}
2581
+ self.temps_count = 0
2582
+
2583
+ def reset(self):
2584
+ for type, cnames in self.temps_allocated.items():
2585
+ self.temps_free[type] = list(cnames)
2586
+
2587
+ def allocate_temp(self, type):
2588
+ if type not in self.temps_allocated:
2589
+ self.temps_allocated[type] = []
2590
+ self.temps_free[type] = []
2591
+ elif self.temps_free[type]:
2592
+ return self.temps_free[type].pop(0)
2593
+ cname = '%s%d' % (Naming.codewriter_temp_prefix, self.temps_count)
2594
+ self.klass.declare_var(pos=None, name=cname, cname=cname, type=type, is_cdef=True)
2595
+ self.temps_allocated[type].append(cname)
2596
+ self.temps_count += 1
2597
+ return cname
Cython/Compiler/CodeGeneration.py ADDED
@@ -0,0 +1,35 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ from .Visitor import VisitorTransform
4
+ from .Nodes import StatListNode
5
+
6
+
7
+ class ExtractPxdCode(VisitorTransform):
8
+ """
9
+ Finds nodes in a pxd file that should generate code, and
10
+ returns them in a StatListNode.
11
+
12
+ The result is a tuple (StatListNode, ModuleScope), i.e.
13
+ everything that is needed from the pxd after it is processed.
14
+
15
+ A purer approach would be to separately compile the pxd code,
16
+ but the result would have to be slightly more sophisticated
17
+ than pure strings (functions + wanted interned strings +
18
+ wanted utility code + wanted cached objects) so for now this
19
+ approach is taken.
20
+ """
21
+
22
+ def __call__(self, root):
23
+ self.funcs = []
24
+ self.visitchildren(root)
25
+ return (StatListNode(root.pos, stats=self.funcs), root.scope)
26
+
27
+ def visit_FuncDefNode(self, node):
28
+ self.funcs.append(node)
29
+ # Do not visit children, nested funcdefnodes will
30
+ # also be moved by this action...
31
+ return node
32
+
33
+ def visit_Node(self, node):
34
+ self.visitchildren(node)
35
+ return node
Cython/Compiler/CythonScope.py ADDED
@@ -0,0 +1,164 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ from .Symtab import ModuleScope
4
+ from .PyrexTypes import *
5
+ from .UtilityCode import CythonUtilityCode
6
+ from .Errors import error
7
+ from .Scanning import StringSourceDescriptor
8
+ from . import MemoryView
9
+
10
+
11
+ class CythonScope(ModuleScope):
12
+ is_cython_builtin = 1
13
+ _cythonscope_initialized = False
14
+
15
+ def __init__(self, context):
16
+ ModuleScope.__init__(self, u'cython', None, None)
17
+ self.pxd_file_loaded = True
18
+ self.populate_cython_scope()
19
+ # The Main.Context object
20
+ self.context = context
21
+
22
+ for fused_type in (cy_integral_type, cy_floating_type, cy_numeric_type):
23
+ entry = self.declare_typedef(fused_type.name,
24
+ fused_type,
25
+ None,
26
+ cname='<error>')
27
+ entry.in_cinclude = True
28
+
29
+ def is_cpp(self):
30
+ # Allow C++ utility code in C++ contexts.
31
+ return self.context.cpp
32
+
33
+ def lookup_type(self, name):
34
+ # This function should go away when types are all first-level objects.
35
+ type = parse_basic_type(name)
36
+ if type:
37
+ return type
38
+
39
+ return super(CythonScope, self).lookup_type(name)
40
+
41
+ def lookup(self, name):
42
+ entry = super(CythonScope, self).lookup(name)
43
+
44
+ if entry is None and not self._cythonscope_initialized:
45
+ self.load_cythonscope()
46
+ entry = super(CythonScope, self).lookup(name)
47
+
48
+ return entry
49
+
50
+ def find_module(self, module_name, pos):
51
+ error("cython.%s is not available" % module_name, pos)
52
+
53
+ def find_submodule(self, module_name):
54
+ entry = self.entries.get(module_name, None)
55
+ if not entry:
56
+ self.load_cythonscope()
57
+ entry = self.entries.get(module_name, None)
58
+
59
+ if entry and entry.as_module:
60
+ return entry.as_module
61
+ else:
62
+ # TODO: fix find_submodule control flow so that we're not
63
+ # expected to create a submodule here (to protect CythonScope's
64
+ # possible immutability). Hack ourselves out of the situation
65
+ # for now.
66
+ raise error((StringSourceDescriptor(u"cython", u""), 0, 0),
67
+ "cython.%s is not available" % module_name)
68
+
69
+ def lookup_qualified_name(self, qname):
70
+ # ExprNode.as_cython_attribute generates qnames and we untangle it here...
71
+ name_path = qname.split(u'.')
72
+ scope = self
73
+ while len(name_path) > 1:
74
+ scope = scope.lookup_here(name_path[0])
75
+ if scope:
76
+ scope = scope.as_module
77
+ del name_path[0]
78
+ if scope is None:
79
+ return None
80
+ else:
81
+ return scope.lookup_here(name_path[0])
82
+
83
+ def populate_cython_scope(self):
84
+ # These are used to optimize isinstance in FinalOptimizePhase
85
+ type_object = self.declare_typedef(
86
+ 'PyTypeObject',
87
+ base_type = c_void_type,
88
+ pos = None,
89
+ cname = 'PyTypeObject')
90
+ type_object.is_void = True
91
+ type_object_type = type_object.type
92
+
93
+ self.declare_cfunction(
94
+ 'PyObject_TypeCheck',
95
+ CFuncType(c_bint_type, [CFuncTypeArg("o", py_object_type, None),
96
+ CFuncTypeArg("t", c_ptr_type(type_object_type), None)]),
97
+ pos = None,
98
+ defining = 1,
99
+ cname = 'PyObject_TypeCheck')
100
+
101
+ def load_cythonscope(self):
102
+ """
103
+ Creates some entries for testing purposes and entries for
104
+ cython.array() and for cython.view.*.
105
+ """
106
+ if self._cythonscope_initialized:
107
+ return
108
+
109
+ self._cythonscope_initialized = True
110
+ cython_testscope_utility_code.declare_in_scope(
111
+ self, cython_scope=self)
112
+ cython_test_extclass_utility_code.declare_in_scope(
113
+ self, cython_scope=self)
114
+
115
+ #
116
+ # The view sub-scope
117
+ #
118
+ self.viewscope = viewscope = ModuleScope(u'view', self, None)
119
+ self.declare_module('view', viewscope, None).as_module = viewscope
120
+ viewscope.is_cython_builtin = True
121
+ viewscope.pxd_file_loaded = True
122
+
123
+ cythonview_testscope_utility_code.declare_in_scope(
124
+ viewscope, cython_scope=self)
125
+
126
+ view_utility_scope = MemoryView.view_utility_code.declare_in_scope(
127
+ self.viewscope, cython_scope=self,
128
+ whitelist=MemoryView.view_utility_whitelist)
129
+
130
+ # self.entries["array"] = view_utility_scope.entries.pop("array")
131
+
132
+
133
+ def create_cython_scope(context):
134
+ # One could in fact probably make it a singleton,
135
+ # but not sure yet whether any code mutates it (which would kill reusing
136
+ # it across different contexts)
137
+ return CythonScope(context)
138
+
139
+ # Load test utilities for the cython scope
140
+
141
+ def load_testscope_utility(cy_util_name, **kwargs):
142
+ return CythonUtilityCode.load(cy_util_name, "TestCythonScope.pyx", **kwargs)
143
+
144
+
145
+ undecorated_methods_protos = UtilityCode(proto=u"""
146
+ /* These methods are undecorated and have therefore no prototype */
147
+ static PyObject *__pyx_TestClass_cdef_method(
148
+ struct __pyx_TestClass_obj *self, int value);
149
+ static PyObject *__pyx_TestClass_cpdef_method(
150
+ struct __pyx_TestClass_obj *self, int value, int skip_dispatch);
151
+ static PyObject *__pyx_TestClass_def_method(
152
+ PyObject *self, PyObject *value);
153
+ """)
154
+
155
+ cython_testscope_utility_code = load_testscope_utility("TestScope")
156
+
157
+ test_cython_utility_dep = load_testscope_utility("TestDep")
158
+
159
+ cython_test_extclass_utility_code = \
160
+ load_testscope_utility("TestClass", name="TestClass",
161
+ requires=[undecorated_methods_protos,
162
+ test_cython_utility_dep])
163
+
164
+ cythonview_testscope_utility_code = load_testscope_utility("View.TestScope")
Cython/Compiler/DebugFlags.py ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Can be enabled at the command line with --debug-xxx.
2
+
3
+ debug_disposal_code = 0
4
+ debug_temp_alloc = 0
5
+ debug_coercion = 0
6
+
7
+ # Write comments into the C code that show where temporary variables
8
+ # are allocated and released.
9
+ debug_temp_code_comments = 0
10
+
11
+ # Write a call trace of the code generation phase into the C code.
12
+ debug_trace_code_generation = 0
13
+
14
+ # Do not replace exceptions with user-friendly error messages.
15
+ debug_no_exception_intercept = 0
16
+
17
+ # Print a message each time a new stage in the pipeline is entered.
18
+ debug_verbose_pipeline = 0
19
+
20
+ # Raise an exception when an error is encountered.
21
+ debug_exception_on_error = 0
Cython/Compiler/Errors.py ADDED
@@ -0,0 +1,265 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #
2
+ # Errors
3
+ #
4
+
5
+ from __future__ import absolute_import
6
+
7
+ try:
8
+ from __builtin__ import basestring as any_string_type
9
+ except ImportError:
10
+ any_string_type = (bytes, str)
11
+
12
+ import sys
13
+ from contextlib import contextmanager
14
+
15
+ from ..Utils import open_new_file
16
+ from . import DebugFlags
17
+ from . import Options
18
+
19
+
20
+ class PyrexError(Exception):
21
+ pass
22
+
23
+
24
+ class PyrexWarning(Exception):
25
+ pass
26
+
27
+
28
+ def context(position):
29
+ source = position[0]
30
+ assert not (isinstance(source, any_string_type)), (
31
+ "Please replace filename strings with Scanning.FileSourceDescriptor instances %r" % source)
32
+ try:
33
+ F = source.get_lines()
34
+ except UnicodeDecodeError:
35
+ # file has an encoding problem
36
+ s = u"[unprintable code]\n"
37
+ else:
38
+ s = u''.join(F[max(0, position[1]-6):position[1]])
39
+ s = u'...\n%s%s^\n' % (s, u' '*(position[2]-1))
40
+ s = u'%s\n%s%s\n' % (u'-'*60, s, u'-'*60)
41
+ return s
42
+
43
+ def format_position(position):
44
+ if position:
45
+ return u"%s:%d:%d: " % (position[0].get_error_description(),
46
+ position[1], position[2])
47
+ return u''
48
+
49
+ def format_error(message, position):
50
+ if position:
51
+ pos_str = format_position(position)
52
+ cont = context(position)
53
+ message = u'\nError compiling Cython file:\n%s\n%s%s' % (cont, pos_str, message or u'')
54
+ return message
55
+
56
+ class CompileError(PyrexError):
57
+
58
+ def __init__(self, position = None, message = u""):
59
+ self.position = position
60
+ self.message_only = message
61
+ self.formatted_message = format_error(message, position)
62
+ self.reported = False
63
+ # Deprecated and withdrawn in 2.6:
64
+ # self.message = message
65
+ Exception.__init__(self, self.formatted_message)
66
+ # Python Exception subclass pickling is broken,
67
+ # see http://bugs.python.org/issue1692335
68
+ self.args = (position, message)
69
+
70
+ def __str__(self):
71
+ return self.formatted_message
72
+
73
+ class CompileWarning(PyrexWarning):
74
+
75
+ def __init__(self, position = None, message = ""):
76
+ self.position = position
77
+ # Deprecated and withdrawn in 2.6:
78
+ # self.message = message
79
+ Exception.__init__(self, format_position(position) + message)
80
+
81
+ class InternalError(Exception):
82
+ # If this is ever raised, there is a bug in the compiler.
83
+
84
+ def __init__(self, message):
85
+ self.message_only = message
86
+ Exception.__init__(self, u"Internal compiler error: %s"
87
+ % message)
88
+
89
+ class AbortError(Exception):
90
+ # Throw this to stop the compilation immediately.
91
+
92
+ def __init__(self, message):
93
+ self.message_only = message
94
+ Exception.__init__(self, u"Abort error: %s" % message)
95
+
96
+ class CompilerCrash(CompileError):
97
+ # raised when an unexpected exception occurs in a transform
98
+ def __init__(self, pos, context, message, cause, stacktrace=None):
99
+ if message:
100
+ message = u'\n' + message
101
+ else:
102
+ message = u'\n'
103
+ self.message_only = message
104
+ if context:
105
+ message = u"Compiler crash in %s%s" % (context, message)
106
+ if stacktrace:
107
+ import traceback
108
+ message += (
109
+ u'\n\nCompiler crash traceback from this point on:\n' +
110
+ u''.join(traceback.format_tb(stacktrace)))
111
+ if cause:
112
+ if not stacktrace:
113
+ message += u'\n'
114
+ message += u'%s: %s' % (cause.__class__.__name__, cause)
115
+ CompileError.__init__(self, pos, message)
116
+ # Python Exception subclass pickling is broken,
117
+ # see http://bugs.python.org/issue1692335
118
+ self.args = (pos, context, message, cause, stacktrace)
119
+
120
+ class NoElementTreeInstalledException(PyrexError):
121
+ """raised when the user enabled options.gdb_debug but no ElementTree
122
+ implementation was found
123
+ """
124
+
125
+ listing_file = None
126
+ num_errors = 0
127
+ echo_file = None
128
+
129
+ def open_listing_file(path, echo_to_stderr = 1):
130
+ # Begin a new error listing. If path is None, no file
131
+ # is opened, the error counter is just reset.
132
+ global listing_file, num_errors, echo_file
133
+ if path is not None:
134
+ listing_file = open_new_file(path)
135
+ else:
136
+ listing_file = None
137
+ if echo_to_stderr:
138
+ echo_file = sys.stderr
139
+ else:
140
+ echo_file = None
141
+ num_errors = 0
142
+
143
+ def close_listing_file():
144
+ global listing_file
145
+ if listing_file:
146
+ listing_file.close()
147
+ listing_file = None
148
+
149
+ def report_error(err, use_stack=True):
150
+ if error_stack and use_stack:
151
+ error_stack[-1].append(err)
152
+ else:
153
+ global num_errors
154
+ # See Main.py for why dual reporting occurs. Quick fix for now.
155
+ if err.reported: return
156
+ err.reported = True
157
+ try: line = u"%s\n" % err
158
+ except UnicodeEncodeError:
159
+ # Python <= 2.5 does this for non-ASCII Unicode exceptions
160
+ line = format_error(getattr(err, 'message_only', "[unprintable exception message]"),
161
+ getattr(err, 'position', None)) + u'\n'
162
+ if listing_file:
163
+ try: listing_file.write(line)
164
+ except UnicodeEncodeError:
165
+ listing_file.write(line.encode('ASCII', 'replace'))
166
+ if echo_file:
167
+ try: echo_file.write(line)
168
+ except UnicodeEncodeError:
169
+ echo_file.write(line.encode('ASCII', 'replace'))
170
+ num_errors += 1
171
+ if Options.fast_fail:
172
+ raise AbortError("fatal errors")
173
+
174
+
175
+ def error(position, message):
176
+ #print("Errors.error:", repr(position), repr(message)) ###
177
+ if position is None:
178
+ raise InternalError(message)
179
+ err = CompileError(position, message)
180
+ if DebugFlags.debug_exception_on_error: raise Exception(err) # debug
181
+ report_error(err)
182
+ return err
183
+
184
+
185
+ LEVEL = 1 # warn about all errors level 1 or higher
186
+
187
+
188
+ def message(position, message, level=1):
189
+ if level < LEVEL:
190
+ return
191
+ warn = CompileWarning(position, message)
192
+ line = "note: %s\n" % warn
193
+ if listing_file:
194
+ listing_file.write(line)
195
+ if echo_file:
196
+ echo_file.write(line)
197
+ return warn
198
+
199
+
200
+ def warning(position, message, level=0):
201
+ if level < LEVEL:
202
+ return
203
+ if Options.warning_errors and position:
204
+ return error(position, message)
205
+ warn = CompileWarning(position, message)
206
+ line = "warning: %s\n" % warn
207
+ if listing_file:
208
+ listing_file.write(line)
209
+ if echo_file:
210
+ echo_file.write(line)
211
+ return warn
212
+
213
+
214
+ _warn_once_seen = {}
215
+ def warn_once(position, message, level=0):
216
+ if level < LEVEL or message in _warn_once_seen:
217
+ return
218
+ warn = CompileWarning(position, message)
219
+ line = "warning: %s\n" % warn
220
+ if listing_file:
221
+ listing_file.write(line)
222
+ if echo_file:
223
+ echo_file.write(line)
224
+ _warn_once_seen[message] = True
225
+ return warn
226
+
227
+
228
+ # These functions can be used to momentarily suppress errors.
229
+
230
+ error_stack = []
231
+
232
+
233
+ def hold_errors():
234
+ error_stack.append([])
235
+
236
+
237
+ def release_errors(ignore=False):
238
+ held_errors = error_stack.pop()
239
+ if not ignore:
240
+ for err in held_errors:
241
+ report_error(err)
242
+
243
+
244
+ def held_errors():
245
+ return error_stack[-1]
246
+
247
+
248
+ # same as context manager:
249
+
250
+ @contextmanager
251
+ def local_errors(ignore=False):
252
+ errors = []
253
+ error_stack.append(errors)
254
+ try:
255
+ yield errors
256
+ finally:
257
+ release_errors(ignore=ignore)
258
+
259
+
260
+ # this module needs a redesign to support parallel cythonisation, but
261
+ # for now, the following works at least in sequential compiler runs
262
+
263
+ def reset():
264
+ _warn_once_seen.clear()
265
+ del error_stack[:]
Cython/Compiler/ExprNodes.py ADDED
The diff for this file is too large to render. See raw diff
 
Cython/Compiler/FlowControl.cp39-win_amd64.pyd ADDED
Binary file (450 kB). View file
 
Cython/Compiler/FlowControl.pxd ADDED
@@ -0,0 +1,111 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ from __future__ import absolute_import
2
+
3
+ cimport cython
4
+
5
+ from .Visitor cimport CythonTransform, TreeVisitor
6
+
7
+ cdef class ControlBlock:
8
+ cdef public set children
9
+ cdef public set parents
10
+ cdef public set positions
11
+ cdef public list stats
12
+ cdef public dict gen
13
+ cdef public set bounded
14
+
15
+ # Big integer bitsets
16
+ cdef public object i_input
17
+ cdef public object i_output
18
+ cdef public object i_gen
19
+ cdef public object i_kill
20
+ cdef public object i_state
21
+
22
+ cpdef bint empty(self)
23
+ cpdef detach(self)
24
+ cpdef add_child(self, block)
25
+
26
+ cdef class ExitBlock(ControlBlock):
27
+ cpdef bint empty(self)
28
+
29
+ cdef class NameAssignment:
30
+ cdef public bint is_arg
31
+ cdef public bint is_deletion
32
+ cdef public object lhs
33
+ cdef public object rhs
34
+ cdef public object entry
35
+ cdef public object pos
36
+ cdef public set refs
37
+ cdef public object bit
38
+ cdef public object inferred_type
39
+
40
+ cdef class AssignmentList:
41
+ cdef public object bit
42
+ cdef public object mask
43
+ cdef public list stats
44
+
45
+ cdef class AssignmentCollector(TreeVisitor):
46
+ cdef list assignments
47
+
48
+ @cython.final
49
+ cdef class ControlFlow:
50
+ cdef public set blocks
51
+ cdef public set entries
52
+ cdef public list loops
53
+ cdef public list exceptions
54
+
55
+ cdef public ControlBlock entry_point
56
+ cdef public ExitBlock exit_point
57
+ cdef public ControlBlock block
58
+
59
+ cdef public dict assmts
60
+
61
+ cpdef newblock(self, ControlBlock parent=*)
62
+ cpdef nextblock(self, ControlBlock parent=*)
63
+ cpdef bint is_tracked(self, entry)
64
+ cpdef bint is_statically_assigned(self, entry)
65
+ cpdef mark_position(self, node)
66
+ cpdef mark_assignment(self, lhs, rhs, entry)
67
+ cpdef mark_argument(self, lhs, rhs, entry)
68
+ cpdef mark_deletion(self, node, entry)
69
+ cpdef mark_reference(self, node, entry)
70
+
71
+ @cython.locals(block=ControlBlock, parent=ControlBlock, unreachable=set)
72
+ cpdef normalize(self)
73
+
74
+ @cython.locals(bit=object, assmts=AssignmentList,
75
+ block=ControlBlock)
76
+ cpdef initialize(self)
77
+
78
+ @cython.locals(assmts=AssignmentList, assmt=NameAssignment)
79
+ cpdef set map_one(self, istate, entry)
80
+
81
+ @cython.locals(block=ControlBlock, parent=ControlBlock)
82
+ cdef reaching_definitions(self)
83
+
84
+ cdef class Uninitialized:
85
+ pass
86
+
87
+ cdef class Unknown:
88
+ pass
89
+
90
+
91
+ cdef class MessageCollection:
92
+ cdef set messages
93
+
94
+
95
+ @cython.locals(dirty=bint, block=ControlBlock, parent=ControlBlock,
96
+ assmt=NameAssignment)
97
+ cdef check_definitions(ControlFlow flow, dict compiler_directives)
98
+
99
+ @cython.final
100
+ cdef class ControlFlowAnalysis(CythonTransform):
101
+ cdef object gv_ctx
102
+ cdef object constant_folder
103
+ cdef set reductions
104
+ cdef list env_stack
105
+ cdef list stack
106
+ cdef object env
107
+ cdef ControlFlow flow
108
+ cdef bint in_inplace_assignment
109
+
110
+ cpdef mark_assignment(self, lhs, rhs=*)
111
+ cpdef mark_position(self, node)