| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define _PY_INTERPRETER
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							|  |  |  | #include "pycore_frame.h"
 | 
					
						
							| 
									
										
										
										
											2023-05-15 20:36:23 -07:00
										 |  |  | #include "pycore_function.h"
 | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | #include "pycore_global_objects.h"
 | 
					
						
							| 
									
										
										
										
											2023-11-14 00:31:02 +00:00
										 |  |  | #include "pycore_compile.h"       // _PyCompile_GetUnaryIntrinsicName, etc
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_intrinsics.h"    // INTRINSIC_PRINT
 | 
					
						
							|  |  |  | #include "pycore_pyerrors.h"      // _PyErr_SetString()
 | 
					
						
							|  |  |  | #include "pycore_runtime.h"       // _Py_ID()
 | 
					
						
							| 
									
										
										
										
											2025-02-26 00:50:26 +02:00
										 |  |  | #include "pycore_sysmodule.h"     // _PySys_GetRequiredAttr()
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_typevarobject.h" // _Py_make_typevar()
 | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 11:54:13 +00:00
										 |  |  | /******** Unary functions ********/ | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | no_intrinsic1(PyThreadState* tstate, PyObject *unused) | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     _PyErr_SetString(tstate, PyExc_SystemError, "invalid intrinsic function"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2025-02-26 00:50:26 +02:00
										 |  |  | print_expr(PyThreadState* Py_UNUSED(ignored), PyObject *value) | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-02-26 00:50:26 +02:00
										 |  |  |     PyObject *hook = _PySys_GetRequiredAttr(&_Py_ID(displayhook)); | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |     // Can't use ERROR_IF here.
 | 
					
						
							|  |  |  |     if (hook == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-02-26 00:50:26 +02:00
										 |  |  |     PyObject *res = PyObject_CallOneArg(hook, value); | 
					
						
							|  |  |  |     Py_DECREF(hook); | 
					
						
							|  |  |  |     return res; | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | import_all_from(PyThreadState *tstate, PyObject *locals, PyObject *v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *all, *dict, *name, *value; | 
					
						
							|  |  |  |     int skip_leading_underscores = 0; | 
					
						
							|  |  |  |     int pos, err; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(v, &_Py_ID(__all__), &all) < 0) { | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |         return -1; /* Unexpected error */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (all == NULL) { | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |         if (PyObject_GetOptionalAttr(v, &_Py_ID(__dict__), &dict) < 0) { | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (dict == NULL) { | 
					
						
							|  |  |  |             _PyErr_SetString(tstate, PyExc_ImportError, | 
					
						
							|  |  |  |                     "from-import-* object has no __dict__ and no __all__"); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         all = PyMapping_Keys(dict); | 
					
						
							|  |  |  |         Py_DECREF(dict); | 
					
						
							|  |  |  |         if (all == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         skip_leading_underscores = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (pos = 0, err = 0; ; pos++) { | 
					
						
							|  |  |  |         name = PySequence_GetItem(all, pos); | 
					
						
							|  |  |  |         if (name == NULL) { | 
					
						
							|  |  |  |             if (!_PyErr_ExceptionMatches(tstate, PyExc_IndexError)) { | 
					
						
							|  |  |  |                 err = -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 _PyErr_Clear(tstate); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!PyUnicode_Check(name)) { | 
					
						
							|  |  |  |             PyObject *modname = PyObject_GetAttr(v, &_Py_ID(__name__)); | 
					
						
							|  |  |  |             if (modname == NULL) { | 
					
						
							|  |  |  |                 Py_DECREF(name); | 
					
						
							|  |  |  |                 err = -1; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (!PyUnicode_Check(modname)) { | 
					
						
							|  |  |  |                 _PyErr_Format(tstate, PyExc_TypeError, | 
					
						
							|  |  |  |                               "module __name__ must be a string, not %.100s", | 
					
						
							|  |  |  |                               Py_TYPE(modname)->tp_name); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 _PyErr_Format(tstate, PyExc_TypeError, | 
					
						
							|  |  |  |                               "%s in %U.%s must be str, not %.100s", | 
					
						
							|  |  |  |                               skip_leading_underscores ? "Key" : "Item", | 
					
						
							|  |  |  |                               modname, | 
					
						
							|  |  |  |                               skip_leading_underscores ? "__dict__" : "__all__", | 
					
						
							|  |  |  |                               Py_TYPE(name)->tp_name); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_DECREF(modname); | 
					
						
							|  |  |  |             Py_DECREF(name); | 
					
						
							|  |  |  |             err = -1; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (skip_leading_underscores) { | 
					
						
							|  |  |  |             if (PyUnicode_READ_CHAR(name, 0) == '_') { | 
					
						
							|  |  |  |                 Py_DECREF(name); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         value = PyObject_GetAttr(v, name); | 
					
						
							|  |  |  |         if (value == NULL) | 
					
						
							|  |  |  |             err = -1; | 
					
						
							|  |  |  |         else if (PyDict_CheckExact(locals)) | 
					
						
							|  |  |  |             err = PyDict_SetItem(locals, name, value); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             err = PyObject_SetItem(locals, name, value); | 
					
						
							|  |  |  |         Py_DECREF(name); | 
					
						
							|  |  |  |         Py_XDECREF(value); | 
					
						
							|  |  |  |         if (err < 0) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_DECREF(all); | 
					
						
							|  |  |  |     return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | import_star(PyThreadState* tstate, PyObject *from) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-17 11:16:03 +01:00
										 |  |  |     _PyInterpreterFrame *frame = tstate->current_frame; | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-04 04:12:10 -07:00
										 |  |  |     PyObject *locals = _PyFrame_GetLocals(frame); | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |     if (locals == NULL) { | 
					
						
							|  |  |  |         _PyErr_SetString(tstate, PyExc_SystemError, | 
					
						
							|  |  |  |                             "no locals found during 'import *'"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int err = import_all_from(tstate, locals, from); | 
					
						
							| 
									
										
										
										
											2024-05-04 04:12:10 -07:00
										 |  |  |     Py_DECREF(locals); | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |     if (err < 0) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | stopiteration_error(PyThreadState* tstate, PyObject *exc) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-17 11:16:03 +01:00
										 |  |  |     _PyInterpreterFrame *frame = tstate->current_frame; | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |     assert(frame->owner == FRAME_OWNED_BY_GENERATOR); | 
					
						
							|  |  |  |     assert(PyExceptionInstance_Check(exc)); | 
					
						
							|  |  |  |     const char *msg = NULL; | 
					
						
							|  |  |  |     if (PyErr_GivenExceptionMatches(exc, PyExc_StopIteration)) { | 
					
						
							|  |  |  |         msg = "generator raised StopIteration"; | 
					
						
							| 
									
										
										
										
											2023-06-14 13:46:37 +01:00
										 |  |  |         if (_PyFrame_GetCode(frame)->co_flags & CO_ASYNC_GENERATOR) { | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |             msg = "async generator raised StopIteration"; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-06-14 13:46:37 +01:00
										 |  |  |         else if (_PyFrame_GetCode(frame)->co_flags & CO_COROUTINE) { | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |             msg = "coroutine raised StopIteration"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-06-14 13:46:37 +01:00
										 |  |  |     else if ((_PyFrame_GetCode(frame)->co_flags & CO_ASYNC_GENERATOR) && | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  |             PyErr_GivenExceptionMatches(exc, PyExc_StopAsyncIteration)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /* code in `gen` raised a StopAsyncIteration error:
 | 
					
						
							|  |  |  |         raise a RuntimeError. | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |         msg = "async generator raised StopAsyncIteration"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (msg != NULL) { | 
					
						
							|  |  |  |         PyObject *message = _PyUnicode_FromASCII(msg, strlen(msg)); | 
					
						
							|  |  |  |         if (message == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyObject *error = PyObject_CallOneArg(PyExc_RuntimeError, message); | 
					
						
							|  |  |  |         if (error == NULL) { | 
					
						
							|  |  |  |             Py_DECREF(message); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         assert(PyExceptionInstance_Check(error)); | 
					
						
							|  |  |  |         PyException_SetCause(error, Py_NewRef(exc)); | 
					
						
							|  |  |  |         // Steal exc reference, rather than Py_NewRef+Py_DECREF
 | 
					
						
							|  |  |  |         PyException_SetContext(error, Py_NewRef(exc)); | 
					
						
							|  |  |  |         Py_DECREF(message); | 
					
						
							|  |  |  |         return error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return Py_NewRef(exc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 14:47:57 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | unary_pos(PyThreadState* unused, PyObject *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return PyNumber_Positive(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | list_to_tuple(PyThreadState* unused, PyObject *v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(PyList_Check(v)); | 
					
						
							|  |  |  |     return _PyTuple_FromArray(((PyListObject *)v)->ob_item, Py_SIZE(v)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 20:36:23 -07:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | make_typevar(PyThreadState* Py_UNUSED(ignored), PyObject *v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(PyUnicode_Check(v)); | 
					
						
							|  |  |  |     return _Py_make_typevar(v, NULL, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define INTRINSIC_FUNC_ENTRY(N, F) \
 | 
					
						
							|  |  |  |     [N] = {F, #N}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const intrinsic_func1_info | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | _PyIntrinsics_UnaryFunctions[] = { | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_1_INVALID, no_intrinsic1) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_PRINT, print_expr) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_IMPORT_STAR, import_star) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_STOPITERATION_ERROR, stopiteration_error) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_ASYNC_GEN_WRAP, _PyAsyncGenValueWrapperNew) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_UNARY_POSITIVE, unary_pos) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_LIST_TO_TUPLE, list_to_tuple) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR, make_typevar) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_PARAMSPEC, _Py_make_paramspec) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVARTUPLE, _Py_make_typevartuple) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_SUBSCRIPT_GENERIC, _Py_subscript_generic) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEALIAS, _Py_make_typealias) | 
					
						
							| 
									
										
										
										
											2023-01-05 16:05:51 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2023-02-14 11:54:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /******** Binary functions ********/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | no_intrinsic2(PyThreadState* tstate, PyObject *unused1, PyObject *unused2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyErr_SetString(tstate, PyExc_SystemError, "invalid intrinsic function"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-02-14 11:54:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | prep_reraise_star(PyThreadState* unused, PyObject *orig, PyObject *excs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(PyList_Check(excs)); | 
					
						
							|  |  |  |     return _PyExc_PrepReraiseStar(orig, excs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 20:36:23 -07:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | make_typevar_with_bound(PyThreadState* Py_UNUSED(ignored), PyObject *name, | 
					
						
							|  |  |  |                         PyObject *evaluate_bound) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(PyUnicode_Check(name)); | 
					
						
							|  |  |  |     return _Py_make_typevar(name, evaluate_bound, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | make_typevar_with_constraints(PyThreadState* Py_UNUSED(ignored), PyObject *name, | 
					
						
							|  |  |  |                               PyObject *evaluate_constraints) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(PyUnicode_Check(name)); | 
					
						
							|  |  |  |     return _Py_make_typevar(name, NULL, evaluate_constraints); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | const intrinsic_func2_info | 
					
						
							| 
									
										
										
										
											2023-02-14 11:54:13 +00:00
										 |  |  | _PyIntrinsics_BinaryFunctions[] = { | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_2_INVALID, no_intrinsic2) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_PREP_RERAISE_STAR, prep_reraise_star) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR_WITH_BOUND, make_typevar_with_bound) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR_WITH_CONSTRAINTS, make_typevar_with_constraints) | 
					
						
							|  |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_SET_FUNCTION_TYPE_PARAMS, _Py_set_function_type_params) | 
					
						
							| 
									
										
										
										
											2024-05-03 06:17:32 -07:00
										 |  |  |     INTRINSIC_FUNC_ENTRY(INTRINSIC_SET_TYPEPARAM_DEFAULT, _Py_set_typeparam_default) | 
					
						
							| 
									
										
										
										
											2023-02-14 11:54:13 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef INTRINSIC_FUNC_ENTRY
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2023-11-14 00:31:02 +00:00
										 |  |  | _PyCompile_GetUnaryIntrinsicName(int index) | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (index < 0 || index > MAX_INTRINSIC_1) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return PyUnicode_FromString(_PyIntrinsics_UnaryFunctions[index].name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2023-11-14 00:31:02 +00:00
										 |  |  | _PyCompile_GetBinaryIntrinsicName(int index) | 
					
						
							| 
									
										
										
										
											2023-07-20 17:46:04 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (index < 0 || index > MAX_INTRINSIC_2) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return PyUnicode_FromString(_PyIntrinsics_BinaryFunctions[index].name); | 
					
						
							|  |  |  | } |