summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Objects/stringlib/string_format.h35
1 files changed, 28 insertions, 7 deletions
diff --git a/Objects/stringlib/string_format.h b/Objects/stringlib/string_format.h
index 2e9c7ef..7e62af5 100644
--- a/Objects/stringlib/string_format.h
+++ b/Objects/stringlib/string_format.h
@@ -483,13 +483,34 @@ render_field(PyObject *fieldobj, SubString *format_spec, OutputString *output)
{
int ok = 0;
PyObject *result = NULL;
+ PyObject *format_spec_object = NULL;
+
+ STRINGLIB_CHAR* format_spec_start = format_spec->ptr ?
+ format_spec->ptr : NULL;
+ Py_ssize_t format_spec_len = format_spec->ptr ?
+ format_spec->end - format_spec->ptr : 0;
+
+ /* If we know the type exactly, skip the lookup of __format__ and just
+ call the formatter directly. */
+ if (PyUnicode_CheckExact(fieldobj))
+ result = _PyUnicode_FormatAdvanced(fieldobj, format_spec_start,
+ format_spec_len);
+ else if (PyLong_CheckExact(fieldobj))
+ result = _PyLong_FormatAdvanced(fieldobj, format_spec_start,
+ format_spec_len);
+ else if (PyFloat_CheckExact(fieldobj))
+ result = _PyFloat_FormatAdvanced(fieldobj, format_spec_start,
+ format_spec_len);
+ else {
+ /* We need to create an object out of the pointers we have, because
+ __format__ takes a string/unicode object for format_spec. */
+ format_spec_object = STRINGLIB_NEW(format_spec_start,
+ format_spec_len);
+ if (format_spec_object == NULL)
+ goto done;
- /* we need to create an object out of the pointers we have */
- PyObject *format_spec_object = SubString_new_object_or_empty(format_spec);
- if (format_spec_object == NULL)
- goto done;
-
- result = PyObject_Format(fieldobj, format_spec_object);
+ result = PyObject_Format(fieldobj, format_spec_object);
+ }
if (result == NULL)
goto done;
@@ -512,7 +533,7 @@ render_field(PyObject *fieldobj, SubString *format_spec, OutputString *output)
ok = output_data(output,
STRINGLIB_STR(result), STRINGLIB_LEN(result));
done:
- Py_DECREF(format_spec_object);
+ Py_XDECREF(format_spec_object);
Py_XDECREF(result);
return ok;
}