summaryrefslogtreecommitdiffstats
path: root/Tools/clinic/clinic.py
diff options
context:
space:
mode:
authorLarry Hastings <larry@hastings.org>2014-01-19 07:50:21 (GMT)
committerLarry Hastings <larry@hastings.org>2014-01-19 07:50:21 (GMT)
commitb7ccb204236dca49f3d8d119aa84631f519add09 (patch)
tree18699632f81936d27c4a4edd1d5346804f5fb466 /Tools/clinic/clinic.py
parentb470575e2492349584d9afa2a9d581b58ee92c38 (diff)
downloadcpython-b7ccb204236dca49f3d8d119aa84631f519add09.zip
cpython-b7ccb204236dca49f3d8d119aa84631f519add09.tar.gz
cpython-b7ccb204236dca49f3d8d119aa84631f519add09.tar.bz2
Issue #20294: Argument Clinic now supports argument parsing for __new__ and
__init__ functions.
Diffstat (limited to 'Tools/clinic/clinic.py')
-rwxr-xr-xTools/clinic/clinic.py482
1 files changed, 294 insertions, 188 deletions
diff --git a/Tools/clinic/clinic.py b/Tools/clinic/clinic.py
index ba0bc17..4d58056 100755
--- a/Tools/clinic/clinic.py
+++ b/Tools/clinic/clinic.py
@@ -169,6 +169,8 @@ def linear_format(s, **kwargs):
themselves. (This line is the "source line".)
* If the substitution text is empty, the source line
is removed in the output.
+ * If the field is not recognized, the original line
+ is passed unmodified through to the output.
* If the substitution text is not empty:
* Each line of the substituted text is indented
by the indent of the source line.
@@ -454,6 +456,182 @@ class CLanguage(Language):
add('"')
return ''.join(text)
+ _templates = {}
+ # the templates will be run through str.format(),
+ # so actual curly-braces need to be doubled up.
+ templates_source = """
+__________________________________________________
+
+docstring_prototype
+
+PyDoc_VAR({c_basename}__doc__);
+__________________________________________________
+
+docstring_definition
+
+PyDoc_STRVAR({c_basename}__doc__,
+{docstring});
+__________________________________________________
+
+impl_definition
+
+static {impl_return_type}
+{c_basename}_impl({impl_parameters})
+__________________________________________________
+
+parser_prototype_noargs
+
+static PyObject *
+{c_basename}({self_type}{self_name}, PyObject *Py_UNUSED(ignored))
+__________________________________________________
+
+parser_prototype_meth_o
+
+# SLIGHT HACK
+# METH_O uses {impl_parameters} for the parser!
+
+static PyObject *
+{c_basename}({impl_parameters})
+__________________________________________________
+
+parser_prototype_varargs
+
+static PyObject *
+{c_basename}({self_type}{self_name}, PyObject *args)
+__________________________________________________
+
+parser_prototype_keyword
+
+static PyObject *
+{c_basename}({self_type}{self_name}, PyObject *args, PyObject *kwargs)
+__________________________________________________
+
+parser_prototype_init
+
+static int
+{c_basename}({self_type}{self_name}, PyObject *args, PyObject *kwargs)
+__________________________________________________
+
+parser_definition_simple_no_parsing
+
+{{
+ return {c_basename}_impl({impl_arguments});
+}}
+__________________________________________________
+
+parser_definition_start
+
+{{
+ {return_value_declaration}
+ {declarations}
+ {initializers}
+{empty line}
+__________________________________________________
+
+parser_definition_end
+
+ {return_conversion}
+
+{exit_label}
+ {cleanup}
+ return return_value;
+}}
+__________________________________________________
+
+parser_definition_impl_call
+
+ {return_value} = {c_basename}_impl({impl_arguments});
+__________________________________________________
+
+parser_definition_unpack_tuple
+
+ if (!PyArg_UnpackTuple(args, "{name}",
+ {unpack_min}, {unpack_max},
+ {parse_arguments}))
+ goto exit;
+__________________________________________________
+
+parser_definition_parse_tuple
+
+ if (!PyArg_ParseTuple(args,
+ "{format_units}:{name}",
+ {parse_arguments}))
+ goto exit;
+__________________________________________________
+
+parser_definition_option_groups
+ {option_group_parsing}
+
+__________________________________________________
+
+parser_definition_parse_tuple_and_keywords
+
+ if (!PyArg_ParseTupleAndKeywords(args, kwargs,
+ "{format_units}:{name}", _keywords,
+ {parse_arguments}))
+ goto exit;
+__________________________________________________
+
+parser_definition_no_positional
+
+ if (!_PyArg_NoPositional("{name}", args))
+ goto exit;
+
+__________________________________________________
+
+parser_definition_no_keywords
+
+ if (!_PyArg_NoKeywords("{name}", kwargs))
+ goto exit;
+
+__________________________________________________
+
+methoddef_define
+
+#define {methoddef_name} \\
+ {{"{name}", (PyCFunction){c_basename}, {methoddef_flags}, {c_basename}__doc__}},
+__________________________________________________
+""".rstrip()
+
+ title = ''
+ buffer = []
+ line = None
+ for line in templates_source.split('\n'):
+ line = line.rstrip()
+ if line.startswith('# '):
+ # comment
+ continue
+ if line.startswith("_____"):
+ if not buffer:
+ continue
+ assert title not in _templates, "defined template twice: " + repr(title)
+ buffer = '\n'.join(buffer).rstrip()
+ buffer = buffer.replace('{empty line}', '')
+ _templates[title] = buffer
+ buffer = []
+ title = ''
+ continue
+ if not title:
+ if not line:
+ continue
+ title = line
+ continue
+ if not (line or buffer):
+ # throw away leading blank lines
+ continue
+ buffer.append(line)
+
+ assert not title, 'ensure templates_source ends with ______ (still adding to ' + repr(title) + ")"
+
+ del templates_source
+ del title
+ del buffer
+ del line
+
+ # for name, value in _templates.items():
+ # print(name + ":")
+ # pprint.pprint(value)
+ # print()
def output_templates(self, f):
parameters = list(f.parameters.values())
@@ -477,12 +655,14 @@ class CLanguage(Language):
else:
all_boring_objects = True
+ new_or_init = f.kind in (METHOD_NEW, METHOD_INIT)
+
meth_o = (len(parameters) == 1 and
parameters[0].kind == inspect.Parameter.POSITIONAL_ONLY and
not converters[0].is_optional() and
isinstance(converters[0], object_converter) and
- converters[0].format_unit == 'O')
-
+ converters[0].format_unit == 'O' and
+ not new_or_init)
# we have to set seven things before we're done:
#
@@ -493,246 +673,144 @@ class CLanguage(Language):
# parser_prototype
# parser_definition
# impl_definition
- #
- # since impl_prototype is always just impl_definition + ';'
- # we just define impl_definition at the top
- docstring_prototype = "PyDoc_VAR({c_basename}__doc__);"
+ templates = self._templates
- docstring_definition = """
-PyDoc_STRVAR({c_basename}__doc__,
-{docstring});
-""".strip()
-
- impl_definition = """
-static {impl_return_type}
-{c_basename}_impl({impl_parameters})""".strip()
+ return_value_declaration = "PyObject *return_value = NULL;"
+ methoddef_define = templates['methoddef_define']
+ docstring_prototype = templates['docstring_prototype']
+ docstring_definition = templates['docstring_definition']
+ impl_definition = templates['impl_definition']
impl_prototype = parser_prototype = parser_definition = None
- def meth_varargs():
- nonlocal flags
- nonlocal parser_prototype
-
- flags = "METH_VARARGS"
+ parser_body_fields = None
+ def parser_body(prototype, *fields):
+ nonlocal parser_body_fields
+ add, output = text_accumulator()
+ add(prototype)
+ parser_body_fields = fields
+ fields = list(fields)
+ fields.insert(0, 'parser_definition_start')
+ fields.append('parser_definition_impl_call')
+ fields.append('parser_definition_end')
+ for field in fields:
+ add('\n')
+ add(templates[field])
+ return output()
- parser_prototype = """
-static PyObject *
-{c_basename}({self_type}{self_name}, PyObject *args)
-""".strip()
+ def insert_keywords(s):
+ return linear_format(s, declarations="static char *_keywords[] = {{{keywords}, NULL}};\n{declarations}")
if not parameters:
# no parameters, METH_NOARGS
flags = "METH_NOARGS"
- parser_prototype = """
-static PyObject *
-{c_basename}({self_type}{self_name}, PyObject *Py_UNUSED(ignored))
-""".strip()
+ parser_prototype = templates['parser_prototype_noargs']
+ parser_definition = parser_prototype
if default_return_converter:
- parser_definition = parser_prototype + """
-{{
- return {c_basename}_impl({impl_arguments});
-}}
-""".rstrip()
+ parser_definition = parser_prototype + '\n' + templates['parser_definition_simple_no_parsing']
else:
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- {declarations}
- {initializers}
-
- {return_value} = {c_basename}_impl({impl_arguments});
- {return_conversion}
-
-{exit_label}
- {cleanup}
- return return_value;
-}}
-""".rstrip()
+ parser_definition = parser_body(parser_prototype)
elif meth_o:
- if default_return_converter:
- # maps perfectly to METH_O, doesn't need a return converter,
- # so we skip the parse function and call
- # directly into the impl function
-
- # SLIGHT HACK
- # METH_O uses {impl_parameters} for the parser.
-
- flags = "METH_O"
-
- impl_definition = """
-static PyObject *
-{c_basename}({impl_parameters})
-""".strip()
+ flags = "METH_O"
+ # impl_definition = templates['parser_prototype_meth_o']
+ if default_return_converter:
+ # maps perfectly to METH_O, doesn't need a return converter.
+ # so we skip making a parse function
+ # and call directly into the impl function.
impl_prototype = parser_prototype = parser_definition = ''
-
+ impl_definition = templates['parser_prototype_meth_o']
else:
- # SLIGHT HACK
- # METH_O uses {impl_parameters} for the parser.
-
- flags = "METH_O"
-
- parser_prototype = """
-static PyObject *
-{c_basename}({impl_parameters})
-""".strip()
-
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- {declarations}
- {initializers}
-
- _return_value = {c_basename}_impl({impl_arguments});
- {return_conversion}
-
-{exit_label}
- {cleanup}
- return return_value;
-}}
-""".rstrip()
+ parser_prototype = templates['parser_prototype_meth_o']
+ parser_definition = parser_body(parser_prototype)
elif has_option_groups:
# positional parameters with option groups
# (we have to generate lots of PyArg_ParseTuple calls
# in a big switch statement)
- meth_varargs()
-
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- {declarations}
- {initializers}
-
- {option_group_parsing}
- {return_value} = {c_basename}_impl({impl_arguments});
- {return_conversion}
+ flags = "METH_VARARGS"
+ parser_prototype = templates['parser_prototype_varargs']
-{exit_label}
- {cleanup}
- return return_value;
-}}
-""".rstrip()
+ parser_definition = parser_body(parser_prototype, 'parser_definition_option_groups')
elif positional and all_boring_objects:
# positional-only, but no option groups,
# and nothing but normal objects:
# PyArg_UnpackTuple!
- meth_varargs()
-
- # substitute in the min and max by hand right here
- assert parameters
- min_o = first_optional
- max_o = len(parameters)
- if isinstance(parameters[0].converter, self_converter):
- min_o -= 1
- max_o -= 1
- min_o = str(min_o)
- max_o = str(max_o)
-
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- {declarations}
- {initializers}
-
- if (!PyArg_UnpackTuple(args, "{name}",
- {min}, {max},
- {parse_arguments}))
- goto exit;
- {return_value} = {c_basename}_impl({impl_arguments});
- {return_conversion}
+ flags = "METH_VARARGS"
+ parser_prototype = templates['parser_prototype_varargs']
-exit:
- {cleanup}
- return return_value;
-}}
-""".rstrip().replace('{min}', min_o).replace('{max}', max_o)
+ parser_definition = parser_body(parser_prototype, 'parser_definition_unpack_tuple')
elif positional:
# positional-only, but no option groups
# we only need one call to PyArg_ParseTuple
- meth_varargs()
-
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- {declarations}
- {initializers}
-
- if (!PyArg_ParseTuple(args,
- "{format_units}:{name}",
- {parse_arguments}))
- goto exit;
- {return_value} = {c_basename}_impl({impl_arguments});
- {return_conversion}
+ flags = "METH_VARARGS"
+ parser_prototype = templates['parser_prototype_varargs']
-exit:
- {cleanup}
- return return_value;
-}}
-""".rstrip()
+ parser_definition = parser_body(parser_prototype, 'parser_definition_parse_tuple')
else:
# positional-or-keyword arguments
flags = "METH_VARARGS|METH_KEYWORDS"
- parser_prototype = """
-static PyObject *
-{c_basename}({self_type}{self_name}, PyObject *args, PyObject *kwargs)
-""".strip()
+ parser_prototype = templates['parser_prototype_keyword']
- parser_definition = parser_prototype + """
-{{
- PyObject *return_value = NULL;
- static char *_keywords[] = {{{keywords}, NULL}};
- {declarations}
- {initializers}
+ parser_definition = parser_body(parser_prototype, 'parser_definition_parse_tuple_and_keywords')
+ parser_definition = insert_keywords(parser_definition)
- if (!PyArg_ParseTupleAndKeywords(args, kwargs,
- "{format_units}:{name}", _keywords,
- {parse_arguments}))
- goto exit;
- {return_value} = {c_basename}_impl({impl_arguments});
- {return_conversion}
-exit:
- {cleanup}
- return return_value;
-}}
-""".rstrip()
+ if new_or_init:
+ methoddef_define = ''
+
+ if f.kind == METHOD_NEW:
+ parser_prototype = templates['parser_prototype_keyword']
+ else:
+ return_value_declaration = "int return_value = -1;"
+ parser_prototype = templates['parser_prototype_init']
+
+ fields = list(parser_body_fields)
+ parses_positional = 'METH_NOARGS' not in flags
+ parses_keywords = 'METH_KEYWORDS' in flags
+ if parses_keywords:
+ assert parses_positional
+
+ if not parses_keywords:
+ fields.insert(0, 'parser_definition_no_keywords')
+ if not parses_positional:
+ fields.insert(0, 'parser_definition_no_positional')
+
+ parser_definition = parser_body(parser_prototype, *fields)
+ if parses_keywords:
+ parser_definition = insert_keywords(parser_definition)
+
if f.methoddef_flags:
- assert flags
flags += '|' + f.methoddef_flags
- methoddef_define = """
-#define {methoddef_name} \\
- {{"{name}", (PyCFunction){c_basename}, {methoddef_flags}, {c_basename}__doc__}},
-""".strip().replace('{methoddef_flags}', flags)
+ methoddef_define = methoddef_define.replace('{methoddef_flags}', flags)
- # parser_prototype mustn't be None, but it could be an empty string.
+ # add ';' to the end of parser_prototype and impl_prototype
+ # (they mustn't be None, but they could be an empty string.)
assert parser_prototype is not None
- assert not parser_prototype.endswith(';')
-
if parser_prototype:
+ assert not parser_prototype.endswith(';')
parser_prototype += ';'
- assert impl_definition
if impl_prototype is None:
- impl_prototype = impl_definition + ";"
+ impl_prototype = impl_definition
+ if impl_prototype:
+ impl_prototype += ";"
- # __new__ and __init__ don't need methoddefs
- if f.kind in (METHOD_NEW, METHOD_INIT):
- methoddef_define = ''
+ parser_definition = parser_definition.replace("{return_value_declaration}", return_value_declaration)
d = {
"docstring_prototype" : docstring_prototype,
@@ -744,8 +822,11 @@ exit:
"impl_definition" : impl_definition,
}
+ # make sure we didn't forget to assign something,
+ # and wrap each non-empty value in \n's
d2 = {}
for name, value in d.items():
+ assert value is not None, "got a None value for template " + repr(name)
if value:
value = '\n' + value + '\n'
d2[name] = value
@@ -881,12 +962,17 @@ exit:
positional = has_option_groups = False
+ first_optional = len(parameters)
+
if parameters:
last_group = 0
- for p in parameters:
+ for i, p in enumerate(parameters):
c = p.converter
+ if p.default is not unspecified:
+ first_optional = min(first_optional, i)
+
# insert group variable
group = p.group
if last_group != group:
@@ -950,6 +1036,10 @@ exit:
template_dict['cleanup'] = "".join(data.cleanup)
template_dict['return_value'] = data.return_value
+ # used by unpack tuple
+ template_dict['unpack_min'] = str(first_optional)
+ template_dict['unpack_max'] = str(len(parameters))
+
if has_option_groups:
self.render_option_group_parsing(f, template_dict)
@@ -2063,7 +2153,7 @@ class char_converter(CConverter):
@add_legacy_c_converter('B', bitwise=True)
-class byte_converter(CConverter):
+class unsigned_char_converter(CConverter):
type = 'unsigned char'
default_type = int
format_unit = 'b'
@@ -2073,6 +2163,8 @@ class byte_converter(CConverter):
if bitwise:
self.format_unit = 'B'
+class byte_converter(unsigned_char_converter): pass
+
class short_converter(CConverter):
type = 'short'
default_type = int
@@ -2455,6 +2547,16 @@ class int_return_converter(long_return_converter):
type = 'int'
cast = '(long)'
+class init_return_converter(long_return_converter):
+ """
+ Special return converter for __init__ functions.
+ """
+ type = 'int'
+ cast = '(long)'
+
+ def render(self, function, data):
+ pass
+
class unsigned_long_return_converter(long_return_converter):
type = 'unsigned long'
conversion_fn = 'PyLong_FromUnsignedLong'
@@ -2858,9 +2960,8 @@ class DSLParser:
if c_basename and not is_legal_c_identifier(c_basename):
fail("Illegal C basename: {}".format(c_basename))
- if not returns:
- return_converter = CReturnConverter()
- else:
+ return_converter = None
+ if returns:
ast_input = "def x() -> {}: pass".format(returns)
module = None
try:
@@ -2893,9 +2994,14 @@ class DSLParser:
if (self.kind != CALLABLE) or (not cls):
fail("__init__ must be a normal method, not a class or static method!")
self.kind = METHOD_INIT
+ if not return_converter:
+ return_converter = init_return_converter()
elif fields[-1] in unsupported_special_methods:
fail(fields[-1] + " should not be converted to Argument Clinic! (Yet.)")
+ if not return_converter:
+ return_converter = CReturnConverter()
+
if not module:
fail("Undefined module used in declaration of " + repr(full_name.strip()) + ".")
self.function = Function(name=function_name, full_name=full_name, module=module, cls=cls, c_basename=c_basename,