Commit 8c975f94 authored by Dag Sverre Seljebotn's avatar Dag Sverre Seljebotn

merge

parents 2fdf96f1 8d1b692a
...@@ -392,13 +392,15 @@ def init_builtins(): ...@@ -392,13 +392,15 @@ def init_builtins():
init_builtin_funcs() init_builtin_funcs()
init_builtin_types() init_builtin_types()
init_builtin_structs() init_builtin_structs()
global list_type, tuple_type, dict_type, set_type, bytes_type, unicode_type, type_type global list_type, tuple_type, dict_type, set_type, type_type
global bytes_type, str_type, unicode_type
type_type = builtin_scope.lookup('type').type type_type = builtin_scope.lookup('type').type
list_type = builtin_scope.lookup('list').type list_type = builtin_scope.lookup('list').type
tuple_type = builtin_scope.lookup('tuple').type tuple_type = builtin_scope.lookup('tuple').type
dict_type = builtin_scope.lookup('dict').type dict_type = builtin_scope.lookup('dict').type
set_type = builtin_scope.lookup('set').type set_type = builtin_scope.lookup('set').type
bytes_type = builtin_scope.lookup('bytes').type bytes_type = builtin_scope.lookup('bytes').type
str_type = builtin_scope.lookup('str').type
unicode_type = builtin_scope.lookup('unicode').type unicode_type = builtin_scope.lookup('unicode').type
init_builtins() init_builtins()
...@@ -289,7 +289,8 @@ class PyObjectConst(object): ...@@ -289,7 +289,8 @@ class PyObjectConst(object):
possible_unicode_identifier = re.compile(ur"(?![0-9])\w+$", re.U).match possible_unicode_identifier = re.compile(ur"(?![0-9])\w+$", re.U).match
possible_bytes_identifier = re.compile(r"(?![0-9])\w+$".encode('ASCII')).match possible_bytes_identifier = re.compile(r"(?![0-9])\w+$".encode('ASCII')).match
nice_identifier = re.compile('^[a-zA-Z0-0_]+$').match nice_identifier = re.compile('^[a-zA-Z0-9_]+$').match
find_alphanums = re.compile('([a-zA-Z0-9]+)').findall
class StringConst(object): class StringConst(object):
"""Global info about a C string constant held by GlobalState. """Global info about a C string constant held by GlobalState.
...@@ -304,19 +305,31 @@ class StringConst(object): ...@@ -304,19 +305,31 @@ class StringConst(object):
self.escaped_value = StringEncoding.escape_byte_string(byte_string) self.escaped_value = StringEncoding.escape_byte_string(byte_string)
self.py_strings = None self.py_strings = None
def get_py_string_const(self, encoding, identifier=None): def get_py_string_const(self, encoding, identifier=None, is_str=False):
py_strings = self.py_strings py_strings = self.py_strings
text = self.text text = self.text
if encoding is not None:
encoding = encoding.upper()
key = (bool(identifier), encoding) is_str = bool(identifier or is_str)
is_unicode = encoding is None and not is_str
if encoding is None:
# unicode string
encoding_key = None
else:
# bytes or str
encoding = encoding.lower()
if encoding in ('utf8', 'utf-8', 'ascii', 'usascii', 'us-ascii'):
encoding = None
encoding_key = None
else:
encoding_key = ''.join(find_alphanums(encoding))
key = (is_str, is_unicode, encoding_key)
if py_strings is not None and key in py_strings: if py_strings is not None and key in py_strings:
py_string = py_strings[key] py_string = py_strings[key]
else: else:
if py_strings is None: if py_strings is None:
self.py_strings = {} self.py_strings = {}
is_unicode = encoding is None
if identifier: if identifier:
intern = True intern = True
elif identifier is None: elif identifier is None:
...@@ -330,14 +343,13 @@ class StringConst(object): ...@@ -330,14 +343,13 @@ class StringConst(object):
prefix = Naming.interned_str_prefix prefix = Naming.interned_str_prefix
else: else:
prefix = Naming.py_const_prefix prefix = Naming.py_const_prefix
pystring_cname = "%s%s%s_%s" % ( pystring_cname = "%s%s_%s" % (
prefix, prefix,
is_unicode and 'u' or 'b', (is_str and 's') or (is_unicode and 'u') or 'b',
identifier and 'i' or '',
self.cname[len(Naming.const_prefix):]) self.cname[len(Naming.const_prefix):])
py_string = PyStringConst( py_string = PyStringConst(
pystring_cname, is_unicode, bool(identifier), intern) pystring_cname, encoding, is_unicode, is_str, intern)
self.py_strings[key] = py_string self.py_strings[key] = py_string
return py_string return py_string
...@@ -346,14 +358,16 @@ class PyStringConst(object): ...@@ -346,14 +358,16 @@ class PyStringConst(object):
"""Global info about a Python string constant held by GlobalState. """Global info about a Python string constant held by GlobalState.
""" """
# cname string # cname string
# unicode boolean # encoding string
# intern boolean # intern boolean
# identifier boolean # is_unicode boolean
# is_str boolean
def __init__(self, cname, is_unicode, identifier=False, intern=False): def __init__(self, cname, encoding, is_unicode, is_str=False, intern=False):
self.cname = cname self.cname = cname
self.identifier = identifier self.encoding = encoding
self.unicode = is_unicode self.is_str = is_str
self.is_unicode = is_unicode
self.intern = intern self.intern = intern
def __lt__(self, other): def __lt__(self, other):
...@@ -393,7 +407,6 @@ class GlobalState(object): ...@@ -393,7 +407,6 @@ class GlobalState(object):
code_layout = [ code_layout = [
'h_code', 'h_code',
'complex_numbers_utility_code',
'utility_code_proto_before_types', 'utility_code_proto_before_types',
'type_declarations', 'type_declarations',
'utility_code_proto', 'utility_code_proto',
...@@ -550,12 +563,15 @@ class GlobalState(object): ...@@ -550,12 +563,15 @@ class GlobalState(object):
c = self.new_string_const(text, byte_string) c = self.new_string_const(text, byte_string)
return c return c
def get_py_string_const(self, text, identifier=None): def get_py_string_const(self, text, identifier=None, is_str=False):
# return a Python string constant, creating a new one if necessary # return a Python string constant, creating a new one if necessary
c_string = self.get_string_const(text) c_string = self.get_string_const(text)
py_string = c_string.get_py_string_const(text.encoding, identifier) py_string = c_string.get_py_string_const(text.encoding, identifier, is_str)
return py_string return py_string
def get_interned_identifier(self, text):
return self.get_py_string_const(text, identifier=True)
def new_string_const(self, text, byte_string): def new_string_const(self, text, byte_string):
cname = self.new_string_const_cname(byte_string) cname = self.new_string_const_cname(byte_string)
c = StringConst(cname, text, byte_string) c = StringConst(cname, text, byte_string)
...@@ -582,7 +598,7 @@ class GlobalState(object): ...@@ -582,7 +598,7 @@ class GlobalState(object):
return self.new_const_cname() return self.new_const_cname()
if len(value) < 20 and nice_identifier(value): if len(value) < 20 and nice_identifier(value):
return "%s%s" % (Naming.const_prefix, value) return "%s_%s" % (Naming.const_prefix, value)
else: else:
return self.new_const_cname() return self.new_const_cname()
...@@ -601,7 +617,7 @@ class GlobalState(object): ...@@ -601,7 +617,7 @@ class GlobalState(object):
def add_cached_builtin_decl(self, entry): def add_cached_builtin_decl(self, entry):
if Options.cache_builtins: if Options.cache_builtins:
if self.should_declare(entry.cname, entry): if self.should_declare(entry.cname, entry):
interned_cname = self.get_py_string_const(entry.name, True).cname interned_cname = self.get_interned_identifier(entry.name).cname
self.put_pyobject_decl(entry) self.put_pyobject_decl(entry)
w = self.parts['cached_builtins'] w = self.parts['cached_builtins']
w.putln('%s = __Pyx_GetName(%s, %s); if (!%s) %s' % ( w.putln('%s = __Pyx_GetName(%s, %s); if (!%s) %s' % (
...@@ -649,18 +665,26 @@ class GlobalState(object): ...@@ -649,18 +665,26 @@ class GlobalState(object):
w.putln("static __Pyx_StringTabEntry %s[] = {" % w.putln("static __Pyx_StringTabEntry %s[] = {" %
Naming.stringtab_cname) Naming.stringtab_cname)
for c_cname, _, py_string in py_strings: for c_cname, _, py_string in py_strings:
if not py_string.is_str or not py_string.encoding or \
py_string.encoding in ('ASCII', 'USASCII', 'US-ASCII',
'UTF8', 'UTF-8'):
encoding = '0'
else:
encoding = '"%s"' % py_string.encoding.lower()
decls_writer.putln( decls_writer.putln(
"static PyObject *%s;" % py_string.cname) "static PyObject *%s;" % py_string.cname)
w.putln( w.putln(
"{&%s, %s, sizeof(%s), %d, %d, %d}," % ( "{&%s, %s, sizeof(%s), %s, %d, %d, %d}," % (
py_string.cname, py_string.cname,
c_cname, c_cname,
c_cname, c_cname,
py_string.unicode, encoding,
py_string.intern, py_string.is_unicode,
py_string.identifier py_string.is_str,
py_string.intern
)) ))
w.putln("{0, 0, 0, 0, 0, 0}") w.putln("{0, 0, 0, 0, 0, 0, 0}")
w.putln("};") w.putln("};")
init_globals = self.parts['init_globals'] init_globals = self.parts['init_globals']
...@@ -894,8 +918,8 @@ class CCodeWriter(object): ...@@ -894,8 +918,8 @@ class CCodeWriter(object):
def get_string_const(self, text): def get_string_const(self, text):
return self.globalstate.get_string_const(text).cname return self.globalstate.get_string_const(text).cname
def get_py_string_const(self, text, identifier=None): def get_py_string_const(self, text, identifier=None, is_str=False):
return self.globalstate.get_py_string_const(text, identifier).cname return self.globalstate.get_py_string_const(text, identifier, is_str).cname
def get_argument_default_const(self, type): def get_argument_default_const(self, type):
return self.globalstate.get_py_const(type).cname return self.globalstate.get_py_const(type).cname
...@@ -904,7 +928,7 @@ class CCodeWriter(object): ...@@ -904,7 +928,7 @@ class CCodeWriter(object):
return self.get_py_string_const(text) return self.get_py_string_const(text)
def intern_identifier(self, text): def intern_identifier(self, text):
return self.get_py_string_const(text, True) return self.get_py_string_const(text, identifier=True)
# code generation # code generation
...@@ -1249,10 +1273,10 @@ class CCodeWriter(object): ...@@ -1249,10 +1273,10 @@ class CCodeWriter(object):
return self.globalstate.lookup_filename(filename) return self.globalstate.lookup_filename(filename)
def put_setup_refcount_context(self, name): def put_setup_refcount_context(self, name):
self.putln('__Pyx_SetupRefcountContext("%s");' % name) self.putln('__Pyx_RefNannySetupContext("%s");' % name)
def put_finish_refcount_context(self): def put_finish_refcount_context(self):
self.putln("__Pyx_FinishRefcountContext();") self.putln("__Pyx_RefNannyFinishContext();")
def put_trace_call(self, name, pos): def put_trace_call(self, name, pos):
self.putln('__Pyx_TraceCall("%s", %s[%s], %s);' % (name, Naming.filetable_cname, self.lookup_filename(pos[0]), pos[1])); self.putln('__Pyx_TraceCall("%s", %s[%s], %s);' % (name, Naming.filetable_cname, self.lookup_filename(pos[0]), pos[1]));
......
...@@ -32,6 +32,22 @@ class NotConstant(object): pass # just for the name ...@@ -32,6 +32,22 @@ class NotConstant(object): pass # just for the name
not_a_constant = NotConstant() not_a_constant = NotConstant()
constant_value_not_set = object() constant_value_not_set = object()
# error messages when coercing from key[0] to key[1]
find_coercion_error = {
# string related errors
(Builtin.unicode_type, Builtin.bytes_type) : "Cannot convert Unicode string to 'bytes' implicitly, encoding required.",
(Builtin.unicode_type, Builtin.str_type) : "Cannot convert Unicode string to 'str' implicitly. This is not portable and requires explicit encoding.",
(Builtin.unicode_type, PyrexTypes.c_char_ptr_type) : "Unicode objects do not support coercion to C types.",
(Builtin.bytes_type, Builtin.unicode_type) : "Cannot convert 'bytes' object to unicode implicitly, decoding required",
(Builtin.bytes_type, Builtin.str_type) : "Cannot convert 'bytes' object to str implicitly. This is not portable to Py3.",
(Builtin.str_type, Builtin.unicode_type) : "str objects do not support coercion to unicode, use a unicode string literal instead (u'')",
(Builtin.str_type, Builtin.bytes_type) : "Cannot convert 'str' to 'bytes' implicitly. This is not portable.",
(Builtin.str_type, PyrexTypes.c_char_ptr_type) : "'str' objects do not support coercion to C types.",
(PyrexTypes.c_char_ptr_type, Builtin.unicode_type) : "Cannot convert 'char*' to unicode implicitly, decoding required",
(PyrexTypes.c_uchar_ptr_type, Builtin.unicode_type) : "Cannot convert 'char*' to unicode implicitly, decoding required",
}.get
class ExprNode(Node): class ExprNode(Node):
# subexprs [string] Class var holding names of subexpr node attrs # subexprs [string] Class var holding names of subexpr node attrs
# type PyrexType Type of the result # type PyrexType Type of the result
...@@ -516,6 +532,9 @@ class ExprNode(Node): ...@@ -516,6 +532,9 @@ class ExprNode(Node):
src_is_py_type = src_type.is_pyobject src_is_py_type = src_type.is_pyobject
dst_is_py_type = dst_type.is_pyobject dst_is_py_type = dst_type.is_pyobject
if self.check_for_coercion_error(dst_type):
return self
if dst_type.is_pyobject: if dst_type.is_pyobject:
if not src.type.is_pyobject: if not src.type.is_pyobject:
src = CoerceToPyTypeNode(src, env) src = CoerceToPyTypeNode(src, env)
...@@ -525,19 +544,32 @@ class ExprNode(Node): ...@@ -525,19 +544,32 @@ class ExprNode(Node):
elif src.type.is_pyobject: elif src.type.is_pyobject:
src = CoerceFromPyTypeNode(dst_type, src, env) src = CoerceFromPyTypeNode(dst_type, src, env)
elif (dst_type.is_complex elif (dst_type.is_complex
and src_type != dst_type and src_type != dst_type
and dst_type.assignable_from(src_type) and dst_type.assignable_from(src_type)):
and not env.directives['c99_complex']):
src = CoerceToComplexNode(src, dst_type, env) src = CoerceToComplexNode(src, dst_type, env)
else: # neither src nor dst are py types else: # neither src nor dst are py types
# Added the string comparison, since for c types that # Added the string comparison, since for c types that
# is enough, but Cython gets confused when the types are # is enough, but Cython gets confused when the types are
# in different pxi files. # in different pxi files.
if not (str(src.type) == str(dst_type) or dst_type.assignable_from(src_type)): if not (str(src.type) == str(dst_type) or dst_type.assignable_from(src_type)):
error(self.pos, "Cannot assign type '%s' to '%s'" % self.fail_assignment(dst_type)
(src.type, dst_type))
return src return src
def fail_assignment(self, dst_type):
error(self.pos, "Cannot assign type '%s' to '%s'" % (self.type, dst_type))
def check_for_coercion_error(self, dst_type, fail=False, default=None):
if fail and not default:
default = "Cannot assign type '%(FROM)s' to '%(TO)s'"
message = find_coercion_error((self.type, dst_type), default)
if message is not None:
error(self.pos, message % {'FROM': self.type, 'TO': dst_type})
return True
if fail:
self.fail_assignment(dst_type)
return True
return False
def coerce_to_pyobject(self, env): def coerce_to_pyobject(self, env):
return self.coerce_to(PyrexTypes.py_object_type, env) return self.coerce_to(PyrexTypes.py_object_type, env)
...@@ -768,6 +800,10 @@ class FloatNode(ConstNode): ...@@ -768,6 +800,10 @@ class FloatNode(ConstNode):
class BytesNode(ConstNode): class BytesNode(ConstNode):
# A char* or bytes literal
#
# value BytesLiteral
type = PyrexTypes.c_char_ptr_type type = PyrexTypes.c_char_ptr_type
def compile_time_value(self, denv): def compile_time_value(self, denv):
...@@ -799,10 +835,19 @@ class BytesNode(ConstNode): ...@@ -799,10 +835,19 @@ class BytesNode(ConstNode):
return self return self
return CharNode(self.pos, value=self.value) return CharNode(self.pos, value=self.value)
if dst_type.is_pyobject and not self.type.is_pyobject: node = self
node = self.as_py_string_node(env) if not self.type.is_pyobject:
else: if dst_type in (py_object_type, Builtin.bytes_type):
node = self node = self.as_py_string_node(env)
elif dst_type.is_pyobject:
self.fail_assignment(dst_type)
return self
else:
node = self
elif dst_type.is_pyobject and dst_type is not py_object_type:
self.check_for_coercion_error(dst_type, fail=True)
return self
# We still need to perform normal coerce_to processing on the # We still need to perform normal coerce_to processing on the
# result, because we might be coercing to an extension type, # result, because we might be coercing to an extension type,
# in which case a type test node will be needed. # in which case a type test node will be needed.
...@@ -832,13 +877,16 @@ class UnicodeNode(PyConstNode): ...@@ -832,13 +877,16 @@ class UnicodeNode(PyConstNode):
# value EncodedString # value EncodedString
type = unicode_type type = unicode_type
def coerce_to(self, dst_type, env): def coerce_to(self, dst_type, env):
if dst_type.is_pyobject: if dst_type is self.type:
return self pass
else: elif not dst_type.is_pyobject:
error(self.pos, "Unicode objects do not support coercion to C types.") error(self.pos, "Unicode objects do not support coercion to C types.")
return self elif dst_type is not py_object_type:
if not self.check_for_coercion_error(dst_type):
self.fail_assignment(dst_type)
return self
def generate_evaluation_code(self, code): def generate_evaluation_code(self, code):
self.result_code = code.get_py_string_const(self.value) self.result_code = code.get_py_string_const(self.value)
...@@ -854,26 +902,34 @@ class StringNode(PyConstNode): ...@@ -854,26 +902,34 @@ class StringNode(PyConstNode):
# A Python str object, i.e. a byte string in Python 2.x and a # A Python str object, i.e. a byte string in Python 2.x and a
# unicode string in Python 3.x # unicode string in Python 3.x
# #
# Can be coerced to a BytesNode (and thus to C types), but not to # value BytesLiteral or EncodedString
# a UnicodeNode. # is_identifier boolean
#
# value BytesLiteral
type = PyrexTypes.py_object_type type = Builtin.str_type
is_identifier = False
def coerce_to(self, dst_type, env): def coerce_to(self, dst_type, env):
if dst_type is Builtin.unicode_type: if dst_type is not py_object_type and dst_type is not Builtin.str_type:
error(self.pos, "str objects do not support coercion to unicode, use a unicode string literal instead (u'')") # if dst_type is Builtin.bytes_type:
return self # # special case: bytes = 'str literal'
if dst_type is Builtin.bytes_type: # return BytesNode(self.pos, value=self.value)
return BytesNode(self.pos, value=self.value) if not dst_type.is_pyobject:
elif dst_type.is_pyobject: return BytesNode(self.pos, value=self.value).coerce_to(dst_type, env)
return self self.check_for_coercion_error(dst_type, fail=True)
else:
return BytesNode(self.pos, value=self.value).coerce_to(dst_type, env) # this will be a unicode string in Py3, so make sure we can decode it
if not self.is_identifier:
encoding = self.value.encoding or 'UTF-8'
try:
self.value.decode(encoding)
except UnicodeDecodeError:
error(self.pos, "String decoding as '%s' failed. Consider using a byte string or unicode string explicitly, or adjust the source code encoding." % encoding)
return self
def generate_evaluation_code(self, code): def generate_evaluation_code(self, code):
self.result_code = code.get_py_string_const(self.value, True) self.result_code = code.get_py_string_const(
self.value, identifier=self.is_identifier, is_str=True)
def get_constant_c_result_code(self): def get_constant_c_result_code(self):
return None return None
...@@ -885,6 +941,12 @@ class StringNode(PyConstNode): ...@@ -885,6 +941,12 @@ class StringNode(PyConstNode):
return self.value return self.value
class IdentifierStringNode(StringNode):
# A special str value that represents an identifier (bytes in Py2,
# unicode in Py3).
is_identifier = True
class LongNode(AtomicExprNode): class LongNode(AtomicExprNode):
# Python long integer literal # Python long integer literal
# #
...@@ -944,8 +1006,6 @@ class ImagNode(AtomicExprNode): ...@@ -944,8 +1006,6 @@ class ImagNode(AtomicExprNode):
def calculate_result_code(self): def calculate_result_code(self):
if self.type.is_pyobject: if self.type.is_pyobject:
return self.result() return self.result()
elif self.c99_complex:
return "%rj" % float(self.value)
else: else:
return "%s(0, %r)" % (self.type.from_parts, float(self.value)) return "%s(0, %r)" % (self.type.from_parts, float(self.value))
...@@ -957,8 +1017,6 @@ class ImagNode(AtomicExprNode): ...@@ -957,8 +1017,6 @@ class ImagNode(AtomicExprNode):
float(self.value), float(self.value),
code.error_goto_if_null(self.result(), self.pos))) code.error_goto_if_null(self.result(), self.pos)))
code.put_gotref(self.py_result()) code.put_gotref(self.py_result())
else:
self.c99_complex = code.globalstate.directives['c99_complex']
...@@ -2951,7 +3009,7 @@ class AttributeNode(ExprNode): ...@@ -2951,7 +3009,7 @@ class AttributeNode(ExprNode):
else: else:
return self.member return self.member
elif obj.type.is_complex: elif obj.type.is_complex:
return "__Pyx_%s_PART(%s)" % (self.member.upper(), obj_code) return "__Pyx_C%s(%s)" % (self.member.upper(), obj_code)
else: else:
return "%s%s%s" % (obj_code, self.op, self.member) return "%s%s%s" % (obj_code, self.op, self.member)
...@@ -4008,7 +4066,7 @@ class UnaryMinusNode(UnopNode): ...@@ -4008,7 +4066,7 @@ class UnaryMinusNode(UnopNode):
else: else:
self.type_error() self.type_error()
if self.type.is_complex: if self.type.is_complex:
self.infix = env.directives['c99_complex'] self.infix = False
def py_operation_function(self): def py_operation_function(self):
return "PyNumber_Negative" return "PyNumber_Negative"
...@@ -4435,7 +4493,7 @@ class NumBinopNode(BinopNode): ...@@ -4435,7 +4493,7 @@ class NumBinopNode(BinopNode):
if not self.type: if not self.type:
self.type_error() self.type_error()
return return
if self.type.is_complex and not env.directives['c99_complex']: if self.type.is_complex:
self.infix = False self.infix = False
if not self.infix: if not self.infix:
self.operand1 = self.operand1.coerce_to(self.type, env) self.operand1 = self.operand1.coerce_to(self.type, env)
...@@ -5083,9 +5141,11 @@ class CmpNode(object): ...@@ -5083,9 +5141,11 @@ class CmpNode(object):
richcmp_constants[op], richcmp_constants[op],
code.error_goto_if_null(result_code, self.pos))) code.error_goto_if_null(result_code, self.pos)))
code.put_gotref(result_code) code.put_gotref(result_code)
elif operand1.type.is_complex and not code.globalstate.directives['c99_complex']: elif operand1.type.is_complex:
if op == "!=": negation = "!" if op == "!=":
else: negation = "" negation = "!"
else:
negation = ""
code.putln("%s = %s(%s%s(%s, %s));" % ( code.putln("%s = %s(%s%s(%s, %s));" % (
result_code, result_code,
coerce_result, coerce_result,
...@@ -5621,8 +5681,8 @@ class CoerceToComplexNode(CoercionNode): ...@@ -5621,8 +5681,8 @@ class CoerceToComplexNode(CoercionNode):
def calculate_result_code(self): def calculate_result_code(self):
if self.arg.type.is_complex: if self.arg.type.is_complex:
real_part = "__Pyx_REAL_PART(%s)" % self.arg.result() real_part = "__Pyx_CREAL(%s)" % self.arg.result()
imag_part = "__Pyx_IMAG_PART(%s)" % self.arg.result() imag_part = "__Pyx_CIMAG(%s)" % self.arg.result()
else: else:
real_part = self.arg.result() real_part = self.arg.result()
imag_part = "0" imag_part = "0"
......
...@@ -260,7 +260,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): ...@@ -260,7 +260,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
globalstate.module_pos = self.pos globalstate.module_pos = self.pos
globalstate.directives = self.directives globalstate.directives = self.directives
globalstate.use_utility_code(refcount_utility_code) globalstate.use_utility_code(refnanny_utility_code)
code = globalstate['before_global_var'] code = globalstate['before_global_var']
code.putln('#define __Pyx_MODULE_NAME "%s"' % self.full_module_name) code.putln('#define __Pyx_MODULE_NAME "%s"' % self.full_module_name)
...@@ -454,6 +454,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): ...@@ -454,6 +454,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code.putln(" #define PyVarObject_HEAD_INIT(type, size) \\") code.putln(" #define PyVarObject_HEAD_INIT(type, size) \\")
code.putln(" PyObject_HEAD_INIT(type) size,") code.putln(" PyObject_HEAD_INIT(type) size,")
code.putln(" #define PyType_Modified(t)") code.putln(" #define PyType_Modified(t)")
code.putln(" #define PyBytes_CheckExact PyString_CheckExact")
code.putln("") code.putln("")
code.putln(" typedef struct {") code.putln(" typedef struct {")
code.putln(" void *buf;") code.putln(" void *buf;")
...@@ -495,7 +496,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): ...@@ -495,7 +496,7 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code.putln("#if PY_MAJOR_VERSION >= 3") code.putln("#if PY_MAJOR_VERSION >= 3")
code.putln(" #define PyBaseString_Type PyUnicode_Type") code.putln(" #define PyBaseString_Type PyUnicode_Type")
code.putln(" #define PyString_Type PyUnicode_Type") code.putln(" #define PyString_Type PyUnicode_Type")
code.putln(" #define PyString_CheckExact PyBytes_CheckExact") code.putln(" #define PyString_CheckExact PyUnicode_CheckExact")
code.putln(" #define PyInt_Type PyLong_Type") code.putln(" #define PyInt_Type PyLong_Type")
code.putln(" #define PyInt_Check(op) PyLong_Check(op)") code.putln(" #define PyInt_Check(op) PyLong_Check(op)")
code.putln(" #define PyInt_CheckExact(op) PyLong_CheckExact(op)") code.putln(" #define PyInt_CheckExact(op) PyLong_CheckExact(op)")
...@@ -558,12 +559,12 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): ...@@ -558,12 +559,12 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code.putln("#include <math.h>") code.putln("#include <math.h>")
code.putln("#define %s" % Naming.api_guard_prefix + self.api_name(env)) code.putln("#define %s" % Naming.api_guard_prefix + self.api_name(env))
self.generate_includes(env, cimported_modules, code) self.generate_includes(env, cimported_modules, code)
if env.directives['c99_complex']: if env.directives['ccomplex']:
code.putln("#ifndef _Complex_I") code.putln("")
code.putln("#include <complex.h>") code.putln("#if !defined(CYTHON_CCOMPLEX)")
code.putln("#define CYTHON_CCOMPLEX 1")
code.putln("#endif") code.putln("#endif")
code.putln("#define __PYX_USE_C99_COMPLEX defined(_Complex_I)") code.putln("")
code.putln('')
code.put(Nodes.utility_function_predeclarations) code.put(Nodes.utility_function_predeclarations)
code.put(PyrexTypes.type_conversion_predeclarations) code.put(PyrexTypes.type_conversion_predeclarations)
code.put(Nodes.branch_prediction_macros) code.put(Nodes.branch_prediction_macros)
...@@ -1633,19 +1634,20 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode): ...@@ -1633,19 +1634,20 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code.putln("{") code.putln("{")
tempdecl_code = code.insertion_point() tempdecl_code = code.insertion_point()
self.generate_filename_init_call(code) code.putln("#if CYTHON_REFNANNY")
code.putln("#ifdef CYTHON_REFNANNY") code.putln("void* __pyx_refnanny = NULL;")
code.putln("void* __pyx_refchk = NULL;") code.putln("__Pyx_RefNanny = __Pyx_RefNannyImportAPI(\"refnanny\");")
code.putln("__Pyx_Refnanny = __Pyx_ImportRefcountAPI(\"refnanny\");") code.putln("if (!__Pyx_RefNanny) {")
code.putln("if (!__Pyx_Refnanny) {")
code.putln(" PyErr_Clear();") code.putln(" PyErr_Clear();")
code.putln(" __Pyx_Refnanny = __Pyx_ImportRefcountAPI(\"Cython.Runtime.refnanny\");") code.putln(" __Pyx_RefNanny = __Pyx_RefNannyImportAPI(\"Cython.Runtime.refnanny\");")
code.putln(" if (!__Pyx_Refnanny)") code.putln(" if (!__Pyx_RefNanny)")
code.putln(" Py_FatalError(\"failed to import refnanny module\");") code.putln(" Py_FatalError(\"failed to import 'refnanny' module\");")
code.putln("}") code.putln("}")
code.putln("__pyx_refchk = __Pyx_Refnanny->NewContext(\"%s\", __LINE__, __FILE__);"% header3) code.putln("__pyx_refnanny = __Pyx_RefNanny->SetupContext(\"%s\", __LINE__, __FILE__);"% header3)
code.putln("#endif") code.putln("#endif")
self.generate_filename_init_call(code)
code.putln("%s = PyTuple_New(0); %s" % (Naming.empty_tuple, code.error_goto_if_null(Naming.empty_tuple, self.pos))); code.putln("%s = PyTuple_New(0); %s" % (Naming.empty_tuple, code.error_goto_if_null(Naming.empty_tuple, self.pos)));
code.putln("#if PY_MAJOR_VERSION < 3"); code.putln("#if PY_MAJOR_VERSION < 3");
code.putln("%s = PyString_FromStringAndSize(\"\", 0); %s" % (Naming.empty_bytes, code.error_goto_if_null(Naming.empty_bytes, self.pos))); code.putln("%s = PyString_FromStringAndSize(\"\", 0); %s" % (Naming.empty_bytes, code.error_goto_if_null(Naming.empty_bytes, self.pos)));
...@@ -2443,36 +2445,51 @@ bad: ...@@ -2443,36 +2445,51 @@ bad:
""" % {'IMPORT_STAR' : Naming.import_star, """ % {'IMPORT_STAR' : Naming.import_star,
'IMPORT_STAR_SET' : Naming.import_star_set } 'IMPORT_STAR_SET' : Naming.import_star_set }
refcount_utility_code = UtilityCode(proto=""" refnanny_utility_code = UtilityCode(proto="""
#ifdef CYTHON_REFNANNY #ifndef CYTHON_REFNANNY
typedef struct { #define CYTHON_REFNANNY 0
void (*INCREF)(void*, PyObject*, int); #endif
void (*DECREF)(void*, PyObject*, int);
void (*GOTREF)(void*, PyObject*, int); #if CYTHON_REFNANNY
void (*GIVEREF)(void*, PyObject*, int); typedef struct {
void* (*NewContext)(const char*, int, const char*); void (*INCREF)(void*, PyObject*, int);
void (*FinishContext)(void**); void (*DECREF)(void*, PyObject*, int);
} __Pyx_RefnannyAPIStruct; void (*GOTREF)(void*, PyObject*, int);
static __Pyx_RefnannyAPIStruct *__Pyx_Refnanny = NULL; void (*GIVEREF)(void*, PyObject*, int);
#define __Pyx_ImportRefcountAPI(name) \ void* (*SetupContext)(const char*, int, const char*);
(__Pyx_RefnannyAPIStruct *) PyCObject_Import((char *)name, (char *)\"RefnannyAPI\") void (*FinishContext)(void**);
#define __Pyx_INCREF(r) __Pyx_Refnanny->INCREF(__pyx_refchk, (PyObject *)(r), __LINE__) } __Pyx_RefNannyAPIStruct;
#define __Pyx_DECREF(r) __Pyx_Refnanny->DECREF(__pyx_refchk, (PyObject *)(r), __LINE__) static __Pyx_RefNannyAPIStruct *__Pyx_RefNanny = NULL;
#define __Pyx_GOTREF(r) __Pyx_Refnanny->GOTREF(__pyx_refchk, (PyObject *)(r), __LINE__) static __Pyx_RefNannyAPIStruct * __Pyx_RefNannyImportAPI(const char *modname) {
#define __Pyx_GIVEREF(r) __Pyx_Refnanny->GIVEREF(__pyx_refchk, (PyObject *)(r), __LINE__) PyObject *m = NULL, *p = NULL;
#define __Pyx_XDECREF(r) do { if((r) != NULL) {__Pyx_DECREF(r);} } while(0) void *r = NULL;
#define __Pyx_SetupRefcountContext(name) \ m = PyImport_ImportModule((char *)modname);
void* __pyx_refchk = __Pyx_Refnanny->NewContext((name), __LINE__, __FILE__) if (!m) goto end;
#define __Pyx_FinishRefcountContext() \ p = PyObject_GetAttrString(m, (char *)\"RefNannyAPI\");
__Pyx_Refnanny->FinishContext(&__pyx_refchk) if (!p) goto end;
r = PyLong_AsVoidPtr(p);
end:
Py_XDECREF(p);
Py_XDECREF(m);
return (__Pyx_RefNannyAPIStruct *)r;
}
#define __Pyx_RefNannySetupContext(name) \
void *__pyx_refnanny = __Pyx_RefNanny->SetupContext((name), __LINE__, __FILE__)
#define __Pyx_RefNannyFinishContext() \
__Pyx_RefNanny->FinishContext(&__pyx_refnanny)
#define __Pyx_INCREF(r) __Pyx_RefNanny->INCREF(__pyx_refnanny, (PyObject *)(r), __LINE__)
#define __Pyx_DECREF(r) __Pyx_RefNanny->DECREF(__pyx_refnanny, (PyObject *)(r), __LINE__)
#define __Pyx_GOTREF(r) __Pyx_RefNanny->GOTREF(__pyx_refnanny, (PyObject *)(r), __LINE__)
#define __Pyx_GIVEREF(r) __Pyx_RefNanny->GIVEREF(__pyx_refnanny, (PyObject *)(r), __LINE__)
#define __Pyx_XDECREF(r) do { if((r) != NULL) {__Pyx_DECREF(r);} } while(0)
#else #else
#define __Pyx_INCREF(r) Py_INCREF(r) #define __Pyx_RefNannySetupContext(name)
#define __Pyx_DECREF(r) Py_DECREF(r) #define __Pyx_RefNannyFinishContext()
#define __Pyx_GOTREF(r) #define __Pyx_INCREF(r) Py_INCREF(r)
#define __Pyx_GIVEREF(r) #define __Pyx_DECREF(r) Py_DECREF(r)
#define __Pyx_XDECREF(r) Py_XDECREF(r) #define __Pyx_GOTREF(r)
#define __Pyx_SetupRefcountContext(name) #define __Pyx_GIVEREF(r)
#define __Pyx_FinishRefcountContext() #define __Pyx_XDECREF(r) Py_XDECREF(r)
#endif /* CYTHON_REFNANNY */ #endif /* CYTHON_REFNANNY */
#define __Pyx_XGIVEREF(r) do { if((r) != NULL) {__Pyx_GIVEREF(r);} } while(0) #define __Pyx_XGIVEREF(r) do { if((r) != NULL) {__Pyx_GIVEREF(r);} } while(0)
#define __Pyx_XGOTREF(r) do { if((r) != NULL) {__Pyx_GOTREF(r);} } while(0) #define __Pyx_XGOTREF(r) do { if((r) != NULL) {__Pyx_GOTREF(r);} } while(0)
......
...@@ -3134,7 +3134,7 @@ class InPlaceAssignmentNode(AssignmentNode): ...@@ -3134,7 +3134,7 @@ class InPlaceAssignmentNode(AssignmentNode):
c_op = "/" c_op = "/"
elif c_op == "**": elif c_op == "**":
error(self.pos, "No C inplace power operator") error(self.pos, "No C inplace power operator")
elif self.lhs.type.is_complex and not code.globalstate.directives['c99_complex']: elif self.lhs.type.is_complex:
error(self.pos, "Inplace operators not implemented for complex types.") error(self.pos, "Inplace operators not implemented for complex types.")
# have to do assignment directly to avoid side-effects # have to do assignment directly to avoid side-effects
...@@ -3479,15 +3479,11 @@ class RaiseStatNode(StatNode): ...@@ -3479,15 +3479,11 @@ class RaiseStatNode(StatNode):
tb_code = self.exc_tb.py_result() tb_code = self.exc_tb.py_result()
else: else:
tb_code = "0" tb_code = "0"
if self.exc_type or self.exc_value or self.exc_tb: code.putln(
code.putln( "__Pyx_Raise(%s, %s, %s);" % (
"__Pyx_Raise(%s, %s, %s);" % ( type_code,
type_code, value_code,
value_code, tb_code))
tb_code))
else:
code.putln(
"__Pyx_ReRaise();")
for obj in (self.exc_type, self.exc_value, self.exc_tb): for obj in (self.exc_type, self.exc_value, self.exc_tb):
if obj: if obj:
obj.generate_disposal_code(code) obj.generate_disposal_code(code)
...@@ -4141,30 +4137,21 @@ class TryExceptStatNode(StatNode): ...@@ -4141,30 +4137,21 @@ class TryExceptStatNode(StatNode):
if error_label_used or not self.has_default_clause: if error_label_used or not self.has_default_clause:
if error_label_used: if error_label_used:
code.put_label(except_error_label) code.put_label(except_error_label)
for var in Naming.exc_save_vars:
code.put_xdecref(var, py_object_type)
code.put_goto(old_error_label)
if code.label_used(try_break_label):
code.put_label(try_break_label)
for var in Naming.exc_save_vars: code.put_xgiveref(var) for var in Naming.exc_save_vars: code.put_xgiveref(var)
code.putln("__Pyx_ExceptionReset(%s);" % code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(Naming.exc_save_vars)) ', '.join(Naming.exc_save_vars))
code.put_goto(old_break_label) code.put_goto(old_error_label)
if code.label_used(try_continue_label):
code.put_label(try_continue_label)
for var in Naming.exc_save_vars: code.put_xgiveref(var)
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(Naming.exc_save_vars))
code.put_goto(old_continue_label)
if code.label_used(except_return_label): for exit_label, old_label in zip(
code.put_label(except_return_label) [try_break_label, try_continue_label, except_return_label],
for var in Naming.exc_save_vars: code.put_xgiveref(var) [old_break_label, old_continue_label, old_return_label]):
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(Naming.exc_save_vars)) if code.label_used(exit_label):
code.put_goto(old_return_label) code.put_label(exit_label)
for var in Naming.exc_save_vars: code.put_xgiveref(var)
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(Naming.exc_save_vars))
code.put_goto(old_label)
if code.label_used(except_end_label): if code.label_used(except_end_label):
code.put_label(except_end_label) code.put_label(except_end_label)
...@@ -4757,7 +4744,7 @@ utility_function_predeclarations = \ ...@@ -4757,7 +4744,7 @@ utility_function_predeclarations = \
#define INLINE #define INLINE
#endif #endif
typedef struct {PyObject **p; char *s; long n; char is_unicode; char intern; char is_identifier;} __Pyx_StringTabEntry; /*proto*/ typedef struct {PyObject **p; char *s; const long n; const char* encoding; const char is_unicode; const char is_str; const char intern; } __Pyx_StringTabEntry; /*proto*/
""" """
...@@ -5006,30 +4993,6 @@ raise_error: ...@@ -5006,30 +4993,6 @@ raise_error:
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
reraise_utility_code = UtilityCode(
proto = """
static void __Pyx_ReRaise(void); /*proto*/
""",
impl = """
static void __Pyx_ReRaise(void) {
PyThreadState *tstate = PyThreadState_GET();
PyObject* tmp_type = tstate->curexc_type;
PyObject* tmp_value = tstate->curexc_value;
PyObject* tmp_tb = tstate->curexc_traceback;
tstate->curexc_type = tstate->exc_type;
tstate->curexc_value = tstate->exc_value;
tstate->curexc_traceback = tstate->exc_traceback;
tstate->exc_type = 0;
tstate->exc_value = 0;
tstate->exc_traceback = 0;
Py_XDECREF(tmp_type);
Py_XDECREF(tmp_value);
Py_XDECREF(tmp_tb);
}
""")
#------------------------------------------------------------------------------------
arg_type_test_utility_code = UtilityCode( arg_type_test_utility_code = UtilityCode(
proto = """ proto = """
static int __Pyx_ArgTypeTest(PyObject *obj, PyTypeObject *type, int none_allowed, static int __Pyx_ArgTypeTest(PyObject *obj, PyTypeObject *type, int none_allowed,
...@@ -5518,7 +5481,7 @@ impl = """ ...@@ -5518,7 +5481,7 @@ impl = """
static int __Pyx_InitStrings(__Pyx_StringTabEntry *t) { static int __Pyx_InitStrings(__Pyx_StringTabEntry *t) {
while (t->p) { while (t->p) {
#if PY_MAJOR_VERSION < 3 #if PY_MAJOR_VERSION < 3
if (t->is_unicode && (!t->is_identifier)) { if (t->is_unicode) {
*t->p = PyUnicode_DecodeUTF8(t->s, t->n - 1, NULL); *t->p = PyUnicode_DecodeUTF8(t->s, t->n - 1, NULL);
} else if (t->intern) { } else if (t->intern) {
*t->p = PyString_InternFromString(t->s); *t->p = PyString_InternFromString(t->s);
...@@ -5526,10 +5489,14 @@ static int __Pyx_InitStrings(__Pyx_StringTabEntry *t) { ...@@ -5526,10 +5489,14 @@ static int __Pyx_InitStrings(__Pyx_StringTabEntry *t) {
*t->p = PyString_FromStringAndSize(t->s, t->n - 1); *t->p = PyString_FromStringAndSize(t->s, t->n - 1);
} }
#else /* Python 3+ has unicode identifiers */ #else /* Python 3+ has unicode identifiers */
if (t->is_identifier || (t->is_unicode && t->intern)) { if (t->is_unicode | t->is_str) {
*t->p = PyUnicode_InternFromString(t->s); if (t->intern) {
} else if (t->is_unicode) { *t->p = PyUnicode_InternFromString(t->s);
*t->p = PyUnicode_FromStringAndSize(t->s, t->n - 1); } else if (t->encoding) {
*t->p = PyUnicode_Decode(t->s, t->n - 1, t->encoding, NULL);
} else {
*t->p = PyUnicode_FromStringAndSize(t->s, t->n - 1);
}
} else { } else {
*t->p = PyBytes_FromStringAndSize(t->s, t->n - 1); *t->p = PyBytes_FromStringAndSize(t->s, t->n - 1);
} }
......
...@@ -65,7 +65,7 @@ option_defaults = { ...@@ -65,7 +65,7 @@ option_defaults = {
'cdivision_warnings': False, 'cdivision_warnings': False,
'always_allow_keywords': False, 'always_allow_keywords': False,
'wraparound' : True, 'wraparound' : True,
'c99_complex' : False, # Don't use macro wrappers for complex arith, not sure what to name this... 'ccomplex' : False, # use C99/C++ for complex types and arith
'callspec' : "", 'callspec' : "",
'profile': False, 'profile': False,
'infer_types': False, 'infer_types': False,
......
...@@ -348,7 +348,7 @@ def p_call(s, function): ...@@ -348,7 +348,7 @@ def p_call(s, function):
s.error("Expected an identifier before '='", s.error("Expected an identifier before '='",
pos = arg.pos) pos = arg.pos)
encoded_name = EncodedString(arg.name) encoded_name = EncodedString(arg.name)
keyword = ExprNodes.StringNode(arg.pos, value = encoded_name) keyword = ExprNodes.IdentifierStringNode(arg.pos, value = encoded_name)
arg = p_simple_expr(s) arg = p_simple_expr(s)
keyword_args.append((keyword, arg)) keyword_args.append((keyword, arg))
else: else:
...@@ -1128,14 +1128,14 @@ def p_import_statement(s): ...@@ -1128,14 +1128,14 @@ def p_import_statement(s):
else: else:
if as_name and "." in dotted_name: if as_name and "." in dotted_name:
name_list = ExprNodes.ListNode(pos, args = [ name_list = ExprNodes.ListNode(pos, args = [
ExprNodes.StringNode(pos, value = EncodedString("*"))]) ExprNodes.IdentifierStringNode(pos, value = EncodedString("*"))])
else: else:
name_list = None name_list = None
stat = Nodes.SingleAssignmentNode(pos, stat = Nodes.SingleAssignmentNode(pos,
lhs = ExprNodes.NameNode(pos, lhs = ExprNodes.NameNode(pos,
name = as_name or target_name), name = as_name or target_name),
rhs = ExprNodes.ImportNode(pos, rhs = ExprNodes.ImportNode(pos,
module_name = ExprNodes.StringNode( module_name = ExprNodes.IdentifierStringNode(
pos, value = dotted_name), pos, value = dotted_name),
name_list = name_list)) name_list = name_list))
stats.append(stat) stats.append(stat)
...@@ -1193,7 +1193,7 @@ def p_from_import_statement(s, first_statement = 0): ...@@ -1193,7 +1193,7 @@ def p_from_import_statement(s, first_statement = 0):
for (name_pos, name, as_name, kind) in imported_names: for (name_pos, name, as_name, kind) in imported_names:
encoded_name = EncodedString(name) encoded_name = EncodedString(name)
imported_name_strings.append( imported_name_strings.append(
ExprNodes.StringNode(name_pos, value = encoded_name)) ExprNodes.IdentifierStringNode(name_pos, value = encoded_name))
items.append( items.append(
(name, (name,
ExprNodes.NameNode(name_pos, ExprNodes.NameNode(name_pos,
...@@ -1203,7 +1203,7 @@ def p_from_import_statement(s, first_statement = 0): ...@@ -1203,7 +1203,7 @@ def p_from_import_statement(s, first_statement = 0):
dotted_name = EncodedString(dotted_name) dotted_name = EncodedString(dotted_name)
return Nodes.FromImportStatNode(pos, return Nodes.FromImportStatNode(pos,
module = ExprNodes.ImportNode(dotted_name_pos, module = ExprNodes.ImportNode(dotted_name_pos,
module_name = ExprNodes.StringNode(pos, value = dotted_name), module_name = ExprNodes.IdentifierStringNode(pos, value = dotted_name),
name_list = import_list), name_list = import_list),
items = items) items = items)
...@@ -1713,7 +1713,7 @@ def p_positional_and_keyword_args(s, end_sy_set, type_positions=(), type_keyword ...@@ -1713,7 +1713,7 @@ def p_positional_and_keyword_args(s, end_sy_set, type_positions=(), type_keyword
parsed_type = True parsed_type = True
else: else:
arg = p_simple_expr(s) arg = p_simple_expr(s)
keyword_node = ExprNodes.StringNode( keyword_node = ExprNodes.IdentifierStringNode(
arg.pos, value = EncodedString(ident)) arg.pos, value = EncodedString(ident))
keyword_args.append((keyword_node, arg)) keyword_args.append((keyword_node, arg))
was_keyword = True was_keyword = True
......
...@@ -823,7 +823,7 @@ class CFloatType(CNumericType): ...@@ -823,7 +823,7 @@ class CFloatType(CNumericType):
class CComplexType(CNumericType): class CComplexType(CNumericType):
is_complex = 1 is_complex = 1
to_py_function = "__pyx_PyObject_from_complex" to_py_function = "__pyx_PyComplex_FromComplex"
has_attributes = 1 has_attributes = 1
scope = None scope = None
...@@ -859,7 +859,9 @@ class CComplexType(CNumericType): ...@@ -859,7 +859,9 @@ class CComplexType(CNumericType):
return self.base_declaration_code(base, entity_code) return self.base_declaration_code(base, entity_code)
def sign_and_name(self): def sign_and_name(self):
return Naming.type_prefix + self.real_type.specalization_name() + "_complex" real_type_name = self.real_type.specalization_name()
real_type_name = real_type_name.replace('long__double','long_double')
return Naming.type_prefix + real_type_name + "_complex"
def assignable_from_resolved_type(self, src_type): def assignable_from_resolved_type(self, src_type):
return (src_type.is_complex and self.real_type.assignable_from_resolved_type(src_type.real_type) return (src_type.is_complex and self.real_type.assignable_from_resolved_type(src_type.real_type)
...@@ -877,21 +879,34 @@ class CComplexType(CNumericType): ...@@ -877,21 +879,34 @@ class CComplexType(CNumericType):
def create_declaration_utility_code(self, env): def create_declaration_utility_code(self, env):
# This must always be run, because a single CComplexType instance can be shared # This must always be run, because a single CComplexType instance can be shared
# across multiple compilations (the one created in the module scope) # across multiple compilations (the one created in the module scope)
env.use_utility_code(complex_generic_utility_code) env.use_utility_code(complex_header_utility_code)
env.use_utility_code( env.use_utility_code(complex_real_imag_utility_code)
complex_arithmatic_utility_code.specialize(self, for utility_code in (complex_type_utility_code,
math_h_modifier = self.real_type.math_h_modifier, complex_from_parts_utility_code,
real_type = self.real_type.declaration_code(''))) complex_arithmatic_utility_code):
env.use_utility_code(
utility_code.specialize(
self,
real_type = self.real_type.declaration_code(''),
m = self.real_type.math_h_modifier))
return True
def create_to_py_utility_code(self, env):
env.use_utility_code(complex_real_imag_utility_code)
env.use_utility_code(complex_to_py_utility_code)
return True return True
def create_from_py_utility_code(self, env): def create_from_py_utility_code(self, env):
self.real_type.create_from_py_utility_code(env) self.real_type.create_from_py_utility_code(env)
env.use_utility_code(
complex_conversion_utility_code.specialize(self, for utility_code in (complex_from_parts_utility_code,
math_h_modifier = self.real_type.math_h_modifier, complex_from_py_utility_code):
real_type = self.real_type.declaration_code(''), env.use_utility_code(
type_convert = self.real_type.from_py_function)) utility_code.specialize(
self.from_py_function = "__pyx_PyObject_As_" + self.specalization_name() self,
real_type = self.real_type.declaration_code(''),
m = self.real_type.math_h_modifier))
self.from_py_function = "__Pyx_PyComplex_As_" + self.specalization_name()
return True return True
def lookup_op(self, nargs, op): def lookup_op(self, nargs, op):
...@@ -901,7 +916,8 @@ class CComplexType(CNumericType): ...@@ -901,7 +916,8 @@ class CComplexType(CNumericType):
pass pass
try: try:
op_name = complex_ops[nargs, op] op_name = complex_ops[nargs, op]
self.binops[nargs, op] = func_name = "%s_%s" % (self.specalization_name(), op_name) modifier = self.real_type.math_h_modifier
self.binops[nargs, op] = func_name = "__Pyx_c_%s%s" % (op_name, modifier)
return func_name return func_name
except KeyError: except KeyError:
return None return None
...@@ -915,114 +931,210 @@ class CComplexType(CNumericType): ...@@ -915,114 +931,210 @@ class CComplexType(CNumericType):
complex_ops = { complex_ops = {
(1, '-'): 'neg', (1, '-'): 'neg',
(1, 'zero'): 'is_zero', (1, 'zero'): 'is_zero',
(2, '+'): 'add', (2, '+'): 'sum',
(2, '-') : 'sub', (2, '-'): 'diff',
(2, '*'): 'mul', (2, '*'): 'prod',
(2, '/'): 'div', (2, '/'): 'quot',
(2, '=='): 'eq', (2, '=='): 'eq',
} }
complex_generic_utility_code = UtilityCode( complex_header_utility_code = UtilityCode(
proto_block='utility_code_proto_before_types',
proto=""" proto="""
#if __PYX_USE_C99_COMPLEX #if !defined(CYTHON_CCOMPLEX)
#define __Pyx_REAL_PART(z) __real__(z) #if defined(__cplusplus)
#define __Pyx_IMAG_PART(z) __imag__(z) #define CYTHON_CCOMPLEX 1
#else #elif defined(_Complex_I)
#define __Pyx_REAL_PART(z) ((z).real) #define CYTHON_CCOMPLEX 1
#define __Pyx_IMAG_PART(z) ((z).imag) #else
#define CYTHON_CCOMPLEX 0
#endif
#endif #endif
#define __pyx_PyObject_from_complex(z) PyComplex_FromDoubles((double)__Pyx_REAL_PART(z), (double)__Pyx_IMAG_PART(z)) #if CYTHON_CCOMPLEX
#ifdef __cplusplus
#include <complex>
#else
#include <complex.h>
#endif
#endif
""") """)
complex_conversion_utility_code = UtilityCode( complex_real_imag_utility_code = UtilityCode(
proto=""" proto="""
static %(type)s __pyx_PyObject_As_%(type_name)s(PyObject* o); /* proto */ #if CYTHON_CCOMPLEX
""", #ifdef __cplusplus
impl=""" #define __Pyx_CREAL(z) ((z).real())
static %(type)s __pyx_PyObject_As_%(type_name)s(PyObject* o) { #define __Pyx_CIMAG(z) ((z).imag())
if (PyComplex_CheckExact(o)) { #else
return %(type_name)s_from_parts( #define __Pyx_CREAL(z) (__real__(z))
(%(real_type)s)((PyComplexObject *)o)->cval.real, #define __Pyx_CIMAG(z) (__imag__(z))
(%(real_type)s)((PyComplexObject *)o)->cval.imag); #endif
} #else
else { #define __Pyx_CREAL(z) ((z).real)
Py_complex cval = PyComplex_AsCComplex(o); #define __Pyx_CIMAG(z) ((z).imag)
return %(type_name)s_from_parts((%(real_type)s)cval.real, (%(real_type)s)cval.imag); #endif
}
}
""") """)
complex_arithmatic_utility_code = UtilityCode( complex_type_utility_code = UtilityCode(
proto_block='utility_code_proto_before_types',
proto=""" proto="""
#if __PYX_USE_C99_COMPLEX #if CYTHON_CCOMPLEX
#ifdef __cplusplus
typedef ::std::complex< %(real_type)s > %(type_name)s;
#else
typedef %(real_type)s _Complex %(type_name)s; typedef %(real_type)s _Complex %(type_name)s;
#endif
#else
typedef struct { %(real_type)s real, imag; } %(type_name)s;
#endif
""")
complex_from_parts_utility_code = UtilityCode(
proto_block='utility_code_proto_before_types',
proto="""
#if CYTHON_CCOMPLEX
#ifdef __cplusplus
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s, %(real_type)s);
#else
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s, %(real_type)s);
#endif
#else
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s, %(real_type)s);
#endif
""",
impl="""
#if CYTHON_CCOMPLEX
#ifdef __cplusplus
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s x, %(real_type)s y) {
return ::std::complex< %(real_type)s >(x, y);
}
#else
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s x, %(real_type)s y) { static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s x, %(real_type)s y) {
return x + y*(%(type)s)_Complex_I; return x + y*(%(type)s)_Complex_I;
} }
#endif
#define %(type_name)s_is_zero(a) ((a) == 0)
#define %(type_name)s_eq(a, b) ((a) == (b))
#define %(type_name)s_add(a, b) ((a)+(b))
#define %(type_name)s_sub(a, b) ((a)-(b))
#define %(type_name)s_mul(a, b) ((a)*(b))
#define %(type_name)s_div(a, b) ((a)/(b))
#define %(type_name)s_neg(a) (-(a))
#else #else
typedef struct { %(real_type)s real, imag; } %(type_name)s;
static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s x, %(real_type)s y) { static INLINE %(type)s %(type_name)s_from_parts(%(real_type)s x, %(real_type)s y) {
%(type)s c; c.real = x; c.imag = y; return c; %(type)s z;
} z.real = x;
z.imag = y;
static INLINE int %(type_name)s_is_zero(%(type)s a) { return z;
return (a.real == 0) & (a.imag == 0);
} }
#endif
""")
static INLINE int %(type_name)s_eq(%(type)s a, %(type)s b) { complex_to_py_utility_code = UtilityCode(
return (a.real == b.real) & (a.imag == b.imag); proto="""
} #define __pyx_PyComplex_FromComplex(z) \\
PyComplex_FromDoubles((double)__Pyx_CREAL(z), \\
(double)__Pyx_CIMAG(z))
""")
complex_from_py_utility_code = UtilityCode(
proto="""
static %(type)s __Pyx_PyComplex_As_%(type_name)s(PyObject*);
""",
impl="""
static %(type)s __Pyx_PyComplex_As_%(type_name)s(PyObject* o) {
Py_complex cval;
if (PyComplex_CheckExact(o))
cval = ((PyComplexObject *)o)->cval;
else
cval = PyComplex_AsCComplex(o);
return %(type_name)s_from_parts(
(%(real_type)s)cval.real,
(%(real_type)s)cval.imag);
}
""")
static INLINE %(type)s %(type_name)s_add(%(type)s a, %(type)s b) { complex_arithmatic_utility_code = UtilityCode(
proto="""
#if CYTHON_CCOMPLEX
#define __Pyx_c_eq%(m)s(a, b) ((a)==(b))
#define __Pyx_c_sum%(m)s(a, b) ((a)+(b))
#define __Pyx_c_diff%(m)s(a, b) ((a)-(b))
#define __Pyx_c_prod%(m)s(a, b) ((a)*(b))
#define __Pyx_c_quot%(m)s(a, b) ((a)/(b))
#define __Pyx_c_neg%(m)s(a) (-(a))
#ifdef __cplusplus
#define __Pyx_c_is_zero%(m)s(z) ((z)==0.0)
#define __Pyx_c_conj%(m)s(z) (::std::conj(z))
/*#define __Pyx_c_abs%(m)s(z) (::std::abs(z))*/
#else
#define __Pyx_c_is_zero%(m)s(z) ((z)==0)
#define __Pyx_c_conj%(m)s(z) (conj%(m)s(z))
/*#define __Pyx_c_abs%(m)s(z) (cabs%(m)s(z))*/
#endif
#else
static INLINE int __Pyx_c_eq%(m)s(%(type)s, %(type)s);
static INLINE %(type)s __Pyx_c_sum%(m)s(%(type)s, %(type)s);
static INLINE %(type)s __Pyx_c_diff%(m)s(%(type)s, %(type)s);
static INLINE %(type)s __Pyx_c_prod%(m)s(%(type)s, %(type)s);
static INLINE %(type)s __Pyx_c_quot%(m)s(%(type)s, %(type)s);
static INLINE %(type)s __Pyx_c_neg%(m)s(%(type)s);
static INLINE int __Pyx_c_is_zero%(m)s(%(type)s);
static INLINE %(type)s __Pyx_c_conj%(m)s(%(type)s);
/*static INLINE %(real_type)s __Pyx_c_abs%(m)s(%(type)s);*/
#endif
""",
impl="""
#if CYTHON_CCOMPLEX
#else
static INLINE int __Pyx_c_eq%(m)s(%(type)s a, %(type)s b) {
return (a.real == b.real) && (a.imag == b.imag);
}
static INLINE %(type)s __Pyx_c_sum%(m)s(%(type)s a, %(type)s b) {
%(type)s z; %(type)s z;
z.real = a.real + b.real; z.real = a.real + b.real;
z.imag = a.imag + b.imag; z.imag = a.imag + b.imag;
return z; return z;
} }
static INLINE %(type)s __Pyx_c_diff%(m)s(%(type)s a, %(type)s b) {
static INLINE %(type)s %(type_name)s_sub(%(type)s a, %(type)s b) {
%(type)s z; %(type)s z;
z.real = a.real - b.real; z.real = a.real - b.real;
z.imag = a.imag - b.imag; z.imag = a.imag - b.imag;
return z; return z;
} }
static INLINE %(type)s __Pyx_c_prod%(m)s(%(type)s a, %(type)s b) {
static INLINE %(type)s %(type_name)s_mul(%(type)s a, %(type)s b) {
%(type)s z; %(type)s z;
z.real = a.real * b.real - a.imag * b.imag; z.real = a.real * b.real - a.imag * b.imag;
z.imag = a.real * b.imag + a.imag * b.real; z.imag = a.real * b.imag + a.imag * b.real;
return z; return z;
} }
static INLINE %(type)s __Pyx_c_quot%(m)s(%(type)s a, %(type)s b) {
static INLINE %(type)s %(type_name)s_div(%(type)s a, %(type)s b) {
%(type)s z; %(type)s z;
%(real_type)s denom = b.real*b.real + b.imag*b.imag; %(real_type)s denom = b.real * b.real + b.imag * b.imag;
z.real = (a.real * b.real + a.imag * b.imag) / denom; z.real = (a.real * b.real + a.imag * b.imag) / denom;
z.imag = (a.imag * b.real - a.real * b.imag) / denom; z.imag = (a.imag * b.real - a.real * b.imag) / denom;
return z; return z;
} }
static INLINE %(type)s __Pyx_c_neg%(m)s(%(type)s a) {
static INLINE %(type)s %(type_name)s_neg(%(type)s a) {
%(type)s z; %(type)s z;
z.real = -a.real; z.real = -a.real;
z.imag = -a.imag; z.imag = -a.imag;
return z; return z;
} }
static INLINE int __Pyx_c_is_zero%(m)s(%(type)s a) {
return (a.real == 0) && (a.imag == 0);
}
static INLINE %(type)s __Pyx_c_conj%(m)s(%(type)s a) {
%(type)s z;
z.real = a.real;
z.imag = -a.imag;
return z;
}
/*
static INLINE %(real_type)s __Pyx_c_abs%(m)s(%(type)s z) {
#if HAVE_HYPOT
return hypot%(m)s(z.real, z.imag);
#else
return sqrt%(m)s(z.real*z.real + z.imag*z.imag);
#endif
}
*/
#endif #endif
""", proto_block='complex_numbers_utility_code') """)
class CArrayType(CType): class CArrayType(CType):
......
...@@ -454,7 +454,6 @@ class PyrexScanner(Scanner): ...@@ -454,7 +454,6 @@ class PyrexScanner(Scanner):
sy = systring sy = systring
else: else:
systring = EncodedString(systring) systring = EncodedString(systring)
systring.encoding = self.source_encoding
self.sy = sy self.sy = sy
self.systring = systring self.systring = systring
if False: # debug_scanner: if False: # debug_scanner:
......
...@@ -1157,7 +1157,7 @@ class StructOrUnionScope(Scope): ...@@ -1157,7 +1157,7 @@ class StructOrUnionScope(Scope):
def declare_cfunction(self, name, type, pos, def declare_cfunction(self, name, type, pos,
cname = None, visibility = 'private', defining = 0, cname = None, visibility = 'private', defining = 0,
api = 0, in_pxd = 0, modifiers = ()): api = 0, in_pxd = 0, modifiers = ()):
self.declare_var(name, type, pos, cname, visibility) return self.declare_var(name, type, pos, cname, visibility)
class ClassScope(Scope): class ClassScope(Scope):
# Abstract base class for namespace of # Abstract base class for namespace of
...@@ -1450,15 +1450,22 @@ static PyObject* __Pyx_Method_ClassMethod(PyObject *method) { ...@@ -1450,15 +1450,22 @@ static PyObject* __Pyx_Method_ClassMethod(PyObject *method) {
} }
if (PyObject_TypeCheck(method, methoddescr_type)) { /* cdef classes */ if (PyObject_TypeCheck(method, methoddescr_type)) { /* cdef classes */
PyMethodDescrObject *descr = (PyMethodDescrObject *)method; PyMethodDescrObject *descr = (PyMethodDescrObject *)method;
return PyDescr_NewClassMethod(descr->d_type, descr->d_method); #if PY_VERSION_HEX < 0x03020000
PyTypeObject *d_type = descr->d_type;
#else
PyTypeObject *d_type = descr->d_common.d_type;
#endif
return PyDescr_NewClassMethod(d_type, descr->d_method);
} }
else if (PyMethod_Check(method)) { /* python classes */ else if (PyMethod_Check(method)) { /* python classes */
return PyClassMethod_New(PyMethod_GET_FUNCTION(method)); return PyClassMethod_New(PyMethod_GET_FUNCTION(method));
} }
else if (PyCFunction_Check(method)) { else if (PyCFunction_Check(method)) {
return PyClassMethod_New(method); return PyClassMethod_New(method);
} }
PyErr_Format(PyExc_TypeError, "Class-level classmethod() can only be called on a method_descriptor or instance method."); PyErr_Format(PyExc_TypeError,
"Class-level classmethod() can only be called on"
"a method_descriptor or instance method.");
return NULL; return NULL;
} }
""") """)
...@@ -125,6 +125,7 @@ from python_mem cimport * ...@@ -125,6 +125,7 @@ from python_mem cimport *
from python_tuple cimport * from python_tuple cimport *
from python_list cimport * from python_list cimport *
from python_object cimport * from python_object cimport *
from python_cobject cimport *
from python_sequence cimport * from python_sequence cimport *
from python_mapping cimport * from python_mapping cimport *
from python_iterator cimport * from python_iterator cimport *
......
cdef extern from "Python.h":
ctypedef void PyObject
###########################################################################
# Warning:
#
# The CObject API is deprecated as of Python 3.1. Please switch to
# the new Capsules API.
###########################################################################
int PyCObject_Check(object p)
# Return true if its argument is a PyCObject.
object PyCObject_FromVoidPtr(void* cobj, void (*destr)(void *))
# Return value: New reference.
#
# Create a PyCObject from the void * cobj. The destr function will
# be called when the object is reclaimed, unless it is NULL.
object PyCObject_FromVoidPtrAndDesc(void* cobj, void* desc, void (*destr)(void *, void *))
# Return value: New reference.
#
# Create a PyCObject from the void * cobj. The destr function will
# be called when the object is reclaimed. The desc argument can be
# used to pass extra callback data for the destructor function.
void* PyCObject_AsVoidPtr(object self)
# Return the object void * that the PyCObject self was created with.
void* PyCObject_GetDesc(object self)
# Return the description void * that the PyCObject self was created with.
int PyCObject_SetVoidPtr(object self, void* cobj)
# Set the void pointer inside self to cobj. The PyCObject must not
# have an associated destructor. Return true on success, false on
# failure.
...@@ -69,7 +69,7 @@ cdef void report_unraisable(object e): ...@@ -69,7 +69,7 @@ cdef void report_unraisable(object e):
# exception has been fetched, in case we are called from # exception has been fetched, in case we are called from
# exception-handling code. # exception-handling code.
cdef PyObject* NewContext(char* funcname, int lineno, char* filename) except NULL: cdef PyObject* SetupContext(char* funcname, int lineno, char* filename) except NULL:
if Context is None: if Context is None:
# Context may be None during finalize phase. # Context may be None during finalize phase.
# In that case, we don't want to be doing anything fancy # In that case, we don't want to be doing anything fancy
...@@ -143,23 +143,23 @@ cdef void FinishContext(PyObject** ctx): ...@@ -143,23 +143,23 @@ cdef void FinishContext(PyObject** ctx):
ctx[0] = NULL ctx[0] = NULL
PyErr_Restore(type, value, tb) PyErr_Restore(type, value, tb)
cdef extern from "Python.h": ctypedef struct RefNannyAPIStruct:
object PyCObject_FromVoidPtr(void*, void (*)(void*))
ctypedef struct RefnannyAPIStruct:
void (*INCREF)(PyObject*, PyObject*, int) void (*INCREF)(PyObject*, PyObject*, int)
void (*DECREF)(PyObject*, PyObject*, int) void (*DECREF)(PyObject*, PyObject*, int)
void (*GOTREF)(PyObject*, PyObject*, int) void (*GOTREF)(PyObject*, PyObject*, int)
void (*GIVEREF)(PyObject*, PyObject*, int) void (*GIVEREF)(PyObject*, PyObject*, int)
PyObject* (*NewContext)(char*, int, char*) except NULL PyObject* (*SetupContext)(char*, int, char*) except NULL
void (*FinishContext)(PyObject**) void (*FinishContext)(PyObject**)
cdef RefnannyAPIStruct api cdef RefNannyAPIStruct api
api.INCREF = INCREF api.INCREF = INCREF
api.DECREF = DECREF api.DECREF = DECREF
api.GOTREF = GOTREF api.GOTREF = GOTREF
api.GIVEREF = GIVEREF api.GIVEREF = GIVEREF
api.NewContext = NewContext api.SetupContext = SetupContext
api.FinishContext = FinishContext api.FinishContext = FinishContext
RefnannyAPI = PyCObject_FromVoidPtr(<void*>&api, NULL) cdef extern from "Python.h":
object PyLong_FromVoidPtr(void*)
RefNannyAPI = PyLong_FromVoidPtr(<void*>&api)
...@@ -59,7 +59,11 @@ class build_ext(_build_ext): ...@@ -59,7 +59,11 @@ class build_ext(_build_ext):
def build_extension(self, ext): def build_extension(self, ext):
if ext.language == 'c++': if ext.language == 'c++':
try: try:
self.compiler.compiler_so.remove('-Wstrict-prototypes') try: # Py2.7+ & Py3.2+
compiler_obj = self.compiler_obj
except AttributeError:
compiler_obj = self.compiler
compiler_obj.compiler_so.remove('-Wstrict-prototypes')
except Exception: except Exception:
pass pass
_build_ext.build_extension(self, ext) _build_ext.build_extension(self, ext)
...@@ -427,9 +431,14 @@ class _FakeClass(object): ...@@ -427,9 +431,14 @@ class _FakeClass(object):
def __init__(self, **kwargs): def __init__(self, **kwargs):
self.__dict__.update(kwargs) self.__dict__.update(kwargs)
class PartialTestResult(unittest._TextTestResult): try: # Py2.7+ and Py3.2+
from unittest.runner import _TextTestResult
except ImportError:
from unittest import _TextTestResult
class PartialTestResult(_TextTestResult):
def __init__(self, base_result): def __init__(self, base_result):
unittest._TextTestResult.__init__( _TextTestResult.__init__(
self, self._StringIO(), True, self, self._StringIO(), True,
base_result.dots + base_result.showAll*2) base_result.dots + base_result.showAll*2)
...@@ -730,7 +739,7 @@ if __name__ == '__main__': ...@@ -730,7 +739,7 @@ if __name__ == '__main__':
build_in_temp=True, build_in_temp=True,
pyxbuild_dir=os.path.join(WORKDIR, "support")) pyxbuild_dir=os.path.join(WORKDIR, "support"))
sys.path.insert(0, os.path.split(libpath)[0]) sys.path.insert(0, os.path.split(libpath)[0])
CFLAGS.append("-DCYTHON_REFNANNY") CFLAGS.append("-DCYTHON_REFNANNY=1")
test_bugs = False test_bugs = False
if options.tickets: if options.tickets:
......
...@@ -12,6 +12,9 @@ cdef double spam "c_spam" (int i, float f): ...@@ -12,6 +12,9 @@ cdef double spam "c_spam" (int i, float f):
cdef foo *p cdef foo *p
global b global b
d = spam(a, f) d = spam(a, f)
cdef foo q
q.i = 7
p = &q
b = p.i b = p.i
p.i = x p.i = x
p.i = y p.i = y
......
# coding: ASCII
# ok:
cdef char* c1 = "abc"
cdef str s1 = "abc"
cdef unicode u1 = u"abc"
cdef bytes b1 = b"abc"
cdef char* c2 = b"abc"
cdef bytes b2 = c1
cdef char* c3 = b1
cdef object o1 = "abc"
cdef object o2 = b"abc"
cdef object o3 = u"abc"
o4 = c1
o5 = b1
o6 = s1
o7 = u1
# errors:
cdef char* c_f1 = u"abc"
cdef char* c_f2 = u1
cdef char* c_f3 = s1
cdef bytes b_f1 = u"abc"
cdef bytes b_f2 = u1
cdef bytes b_f3 = s1
cdef str s_f1 = b"abc"
cdef str s_f2 = b1
cdef str s_f3 = u"abc"
cdef str s_f4 = u1
cdef unicode u_f1 = "abc"
cdef unicode u_f2 = s1
cdef unicode u_f3 = b"abc"
cdef unicode u_f4 = b1
cdef unicode u_f5 = c1
cdef tuple t_f1 = "abc"
cdef tuple t_f2 = u"abc"
cdef tuple t_f3 = b"abc"
cdef list l_f1 = s1
cdef list l_f2 = b1
cdef list l_f3 = u1
_ERRORS = u"""
25:20: Unicode objects do not support coercion to C types.
26:22: Unicode objects do not support coercion to C types.
27:22: 'str' objects do not support coercion to C types.
29:20: Cannot convert Unicode string to 'bytes' implicitly, encoding required.
30:22: Cannot convert Unicode string to 'bytes' implicitly, encoding required.
31:22: Cannot convert 'str' to 'bytes' implicitly. This is not portable.
33:17: Cannot assign type 'char *' to 'str object'
34:19: Cannot convert 'bytes' object to str implicitly. This is not portable to Py3.
35:17: Cannot convert Unicode string to 'str' implicitly. This is not portable and requires explicit encoding.
36:19: Cannot convert Unicode string to 'str' implicitly. This is not portable and requires explicit encoding.
38:20: str objects do not support coercion to unicode, use a unicode string literal instead (u'')
39:22: str objects do not support coercion to unicode, use a unicode string literal instead (u'')
40:20: Cannot assign type 'char *' to 'unicode object'
41:22: Cannot convert 'bytes' object to unicode implicitly, decoding required
42:22: Cannot convert 'char*' to unicode implicitly, decoding required
44:19: Cannot assign type 'str object' to 'tuple object'
45:18: Cannot assign type 'unicode object' to 'tuple object'
46:18: Cannot assign type 'char *' to 'tuple object'
"""
__doc__ = u""" __doc__ = u"""
>>> test_object_conversion(2) >>> test_object_conversion(2)
((2+0j), (2+0j)) ((2+0j), (2+0j), (2+0j))
>>> test_object_conversion(2j - 0.5) >>> test_object_conversion(2j - 0.5)
((-0.5+2j), (-0.5+2j)) ((-0.5+2j), (-0.5+2j), (-0.5+2j))
>>> test_arithmetic(2j, 4j) >>> test_arithmetic(2j, 4j)
(-2j, 6j, -2j, (-8+0j), (0.5+0j)) (2j, -2j, 6j, -2j, (-8+0j), (0.5+0j))
>>> test_arithmetic(6+12j, 3j) >>> test_arithmetic(6+12j, 3j)
((-6-12j), (6+15j), (6+9j), (-36+18j), (4-2j)) ((6+12j), (-6-12j), (6+15j), (6+9j), (-36+18j), (4-2j))
>>> test_arithmetic(5-10j, 3+4j) >>> test_arithmetic(5-10j, 3+4j)
((-5+10j), (8-6j), (2-14j), (55-10j), (-1-2j)) ((5-10j), (-5+10j), (8-6j), (2-14j), (55-10j), (-1-2j))
>>> test_div_by_zero(4j) ## XXX this is not working
-0.25j ## >>> test_div_by_zero(4j)
>>> test_div_by_zero(0) ## -0.25j
Traceback (most recent call last): ## >>> test_div_by_zero(0)
... ## Traceback (most recent call last):
ZeroDivisionError: float division ## ...
## ZeroDivisionError: float division
>>> test_coercion(1, 1.5, 2.5, 4+1j, 10j) >>> test_coercion(1, 1.5, 2.5, 4+1j, 10j)
(1+0j) (1+0j)
...@@ -56,6 +57,10 @@ __doc__ = u""" ...@@ -56,6 +57,10 @@ __doc__ = u"""
(1+2j) (1+2j)
>>> test_real_imag_assignment(1.5, -3.5) >>> test_real_imag_assignment(1.5, -3.5)
(1.5-3.5j) (1.5-3.5j)
## XXX not implemented yet!
## >>> test_conjugate(1+2j)
## (1-2j)
""" """
#cdef extern from "complex.h": #cdef extern from "complex.h":
...@@ -65,15 +70,17 @@ cimport cython ...@@ -65,15 +70,17 @@ cimport cython
def test_object_conversion(o): def test_object_conversion(o):
cdef float complex a = o cdef float complex a = o
cdef double complex z = o cdef double complex b = o
return (a, z) cdef long double complex c = o
return (a, b, c)
def test_arithmetic(double complex z, double complex w): def test_arithmetic(double complex z, double complex w):
return -z, z+w, z-w, z*w, z/w return +z, -z, z+w, z-w, z*w, z/w
@cython.cdivision(False) ## XXX this is not working
def test_div_by_zero(double complex z): ## @cython.cdivision(False)
return 1/z ## def test_div_by_zero(double complex z):
## return 1/z
def test_coercion(int a, float b, double c, float complex d, double complex e): def test_coercion(int a, float b, double c, float complex d, double complex e):
cdef double complex z cdef double complex z
...@@ -102,3 +109,6 @@ def test_real_imag_assignment(object a, double b): ...@@ -102,3 +109,6 @@ def test_real_imag_assignment(object a, double b):
z.imag = b z.imag = b
return z return z
## XXX not implemented yet!
## def test_conjugate(float complex z):
## return z.conjugate()
cdef extern from "complex_numbers_c89_T398.h": pass
include "complex_numbers_T305.pyx"
#if !defined(__cplusplus)
#if (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) \
|| defined(__GNUC__) \
|| defined(__INTEL_COMPILER) \
|| defined(__IBMC__) \
#include <complex.h>
#if !defined(_Complex_I)
#error The "complex.h" header does not define the '_Complex_I' macro.
#error Please report this to Cython developers <cython-dev@codespeak.net>
#endif
#endif
#endif
cdef extern from "complex_numbers_c99_T398.h": pass
include "complex_numbers_T305.pyx"
#if defined(__cplusplus)
#define CYTHON_CCOMPLEX 1
#else
#define CYTHON_CCOMPLEX 0
#endif
cdef extern from "complex_numbers_cxx_T398.h": pass
include "complex_numbers_T305.pyx"
...@@ -68,10 +68,10 @@ __doc__ = ur""" ...@@ -68,10 +68,10 @@ __doc__ = ur"""
'with_doc_4(int a, str b, list c) -> str\n\n Existing string\n ' 'with_doc_4(int a, str b, list c) -> str\n\n Existing string\n '
>>> f_sd.__doc__ >>> f_sd.__doc__
"f_sd(str s=u'spam')" "f_sd(str s='spam')"
>>> cf_sd.__doc__ >>> cf_sd.__doc__
"cf_sd(str s=u'spam') -> str" "cf_sd(str s='spam') -> str"
>>> types.__doc__ >>> types.__doc__
'types(Ext a, int b, unsigned short c, float d, e)' 'types(Ext a, int b, unsigned short c, float d, e)'
...@@ -219,10 +219,10 @@ cpdef str with_doc_4(int a, str b, list c): ...@@ -219,10 +219,10 @@ cpdef str with_doc_4(int a, str b, list c):
""" """
return b return b
def f_sd(str s=u'spam'): def f_sd(str s='spam'):
return s return s
cpdef str cf_sd(str s=u'spam'): cpdef str cf_sd(str s='spam'):
return s return s
cpdef char f_c(char c): cpdef char f_c(char c):
......
...@@ -46,8 +46,6 @@ True ...@@ -46,8 +46,6 @@ True
import sys import sys
IS_PY3 = sys.version_info[0] >= 3
cdef class cy_iterator(object): cdef class cy_iterator(object):
def __iter__(self): def __iter__(self):
return self return self
......
__doc__ = u"""
>>> def bar():
... try:
... foo()
... except ValueError:
... pass
>>> bar()
>>> print(sys.exc_info())
(None, None, None)
"""
import sys
def foo():
try:
raise TypeError
except TypeError:
raise ValueError
...@@ -37,7 +37,7 @@ double quoted string.""" ...@@ -37,7 +37,7 @@ double quoted string."""
i = 'This string\ i = 'This string\
has an ignored newline.' has an ignored newline.'
j = 'One-char escapes: \'\"\\\a\b\f\n\r\t\v' j = 'One-char escapes: \'\"\\\a\b\f\n\r\t\v'
k = 'Oct and hex escapes: \1 \12 \123 \x45 \xaf \xAF' k = b'Oct and hex escapes: \1 \12 \123 \x45 \xaf \xAF'
l = r'''This is\ l = r'''This is\
a \three \line a \three \line
raw string with some backslashes.''' raw string with some backslashes.'''
......
# -*- coding: latin-1 -*-
__doc__ = (u"""
>>> a == 'abc'
True
>>> isinstance(a, str)
True
>>> isinstance(s, str)
True
>>> len(s)
6
>>> s == 'ao'
True
>>> isinstance(add(), str)
True
>>> len(add())
9
>>> add() == 'abcao'
True
>>> isinstance(add_literal(), str)
True
>>> len(add_literal())
9
>>> add_literal() == 'abcao'
True
>>> isinstance(typed(), str)
True
>>> len(typed())
6
>>> typed() == ''
True
"""
# recoding/escaping is required to properly pass the literals to doctest
).encode('unicode_escape').decode('ASCII')
a = 'abc'
s = 'ao'
u = u'ao'
cdef str S = ''
def add():
return a+s
def add_literal():
return 'abc' + 'ao'
def typed():
return S
...@@ -23,7 +23,7 @@ def simple(): ...@@ -23,7 +23,7 @@ def simple():
b = b"abc" b = b"abc"
assert typeof(b) == "char *", typeof(b) assert typeof(b) == "char *", typeof(b)
s = "abc" s = "abc"
assert typeof(s) == "Python object", typeof(s) assert typeof(s) == "str object", typeof(s)
u = u"xyz" u = u"xyz"
assert typeof(u) == "unicode object", typeof(u) assert typeof(u) == "unicode object", typeof(u)
L = [1,2,3] L = [1,2,3]
......
...@@ -52,13 +52,13 @@ def slice_list_assign(list l, value): ...@@ -52,13 +52,13 @@ def slice_list_assign(list l, value):
def slice_charp(py_string_arg): def slice_charp(py_string_arg):
cdef str py_string = py_string_arg.encode(u'ASCII') cdef bytes py_string = py_string_arg.encode(u'ASCII')
cdef char* s = py_string cdef char* s = py_string
return s[1:3].decode(u'ASCII') return s[1:3].decode(u'ASCII')
def slice_charp_repeat(py_string_arg): def slice_charp_repeat(py_string_arg):
cdef str py_string = py_string_arg.encode(u'ASCII') cdef bytes py_string = py_string_arg.encode(u'ASCII')
cdef char* s = py_string cdef char* s = py_string
cdef str slice_val = s[1:6] cdef bytes slice_val = s[1:6]
s = slice_val s = slice_val
return s[1:3].decode(u'ASCII') return s[1:3].decode(u'ASCII')
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment