mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 07:31:38 +00:00 
			
		
		
		
	
		
			
	
	
		
			82 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			82 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								#ifndef Py_EMSCRIPTEN_TRAMPOLINE_H
							 | 
						||
| 
								 | 
							
								#define Py_EMSCRIPTEN_TRAMPOLINE_H
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "pycore_runtime.h"  // _PyRuntimeState
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * C function call trampolines to mitigate bad function pointer casts.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Section 6.3.2.3, paragraph 8 reads:
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *      A pointer to a function of one type may be converted to a pointer to a
							 | 
						||
| 
								 | 
							
								 *      function of another type and back again; the result shall compare equal to
							 | 
						||
| 
								 | 
							
								 *      the original pointer. If a converted pointer is used to call a function
							 | 
						||
| 
								 | 
							
								 *      whose type is not compatible with the pointed-to type, the behavior is
							 | 
						||
| 
								 | 
							
								 *      undefined.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Typical native ABIs ignore additional arguments or fill in missing values
							 | 
						||
| 
								 | 
							
								 * with 0/NULL in function pointer cast. Compilers do not show warnings when a
							 | 
						||
| 
								 | 
							
								 * function pointer is explicitly casted to an incompatible type.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Bad fpcasts are an issue in WebAssembly. WASM's indirect_call has strict
							 | 
						||
| 
								 | 
							
								 * function signature checks. Argument count, types, and return type must match.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Third party code unintentionally rely on problematic fpcasts. The call
							 | 
						||
| 
								 | 
							
								 * trampoline mitigates common occurrences of bad fpcasts on Emscripten.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if defined(__EMSCRIPTEN__) && defined(PY_CALL_TRAMPOLINE)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void _Py_EmscriptenTrampoline_Init(_PyRuntimeState *runtime);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								PyObject*
							 | 
						||
| 
								 | 
							
								_PyEM_TrampolineCall_JavaScript(PyCFunctionWithKeywords func,
							 | 
						||
| 
								 | 
							
								                                PyObject* self,
							 | 
						||
| 
								 | 
							
								                                PyObject* args,
							 | 
						||
| 
								 | 
							
								                                PyObject* kw);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								PyObject*
							 | 
						||
| 
								 | 
							
								_PyEM_TrampolineCall_Reflection(PyCFunctionWithKeywords func,
							 | 
						||
| 
								 | 
							
								                                PyObject* self,
							 | 
						||
| 
								 | 
							
								                                PyObject* args,
							 | 
						||
| 
								 | 
							
								                                PyObject* kw);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _PyEM_TrampolineCall(meth, self, args, kw) \
							 | 
						||
| 
								 | 
							
								    ((_PyRuntime.wasm_type_reflection_available) ? \
							 | 
						||
| 
								 | 
							
								        (_PyEM_TrampolineCall_Reflection((PyCFunctionWithKeywords)(meth), (self), (args), (kw))) : \
							 | 
						||
| 
								 | 
							
								        (_PyEM_TrampolineCall_JavaScript((PyCFunctionWithKeywords)(meth), (self), (args), (kw))))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _PyCFunction_TrampolineCall(meth, self, args) \
							 | 
						||
| 
								 | 
							
								    _PyEM_TrampolineCall( \
							 | 
						||
| 
								 | 
							
								        (*(PyCFunctionWithKeywords)(void(*)(void))(meth)), (self), (args), NULL)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _PyCFunctionWithKeywords_TrampolineCall(meth, self, args, kw) \
							 | 
						||
| 
								 | 
							
								    _PyEM_TrampolineCall((meth), (self), (args), (kw))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define descr_set_trampoline_call(set, obj, value, closure) \
							 | 
						||
| 
								 | 
							
								    ((int)_PyEM_TrampolineCall((PyCFunctionWithKeywords)(set), (obj), (value), (PyObject*)(closure)))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define descr_get_trampoline_call(get, obj, closure) \
							 | 
						||
| 
								 | 
							
								    _PyEM_TrampolineCall((PyCFunctionWithKeywords)(get), (obj), (PyObject*)(closure), NULL)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#else // defined(__EMSCRIPTEN__) && defined(PY_CALL_TRAMPOLINE)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _Py_EmscriptenTrampoline_Init(runtime)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _PyCFunction_TrampolineCall(meth, self, args) \
							 | 
						||
| 
								 | 
							
								    (meth)((self), (args))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define _PyCFunctionWithKeywords_TrampolineCall(meth, self, args, kw) \
							 | 
						||
| 
								 | 
							
								    (meth)((self), (args), (kw))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define descr_set_trampoline_call(set, obj, value, closure) \
							 | 
						||
| 
								 | 
							
								    (set)((obj), (value), (closure))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define descr_get_trampoline_call(get, obj, closure) \
							 | 
						||
| 
								 | 
							
								    (get)((obj), (closure))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif // defined(__EMSCRIPTEN__) && defined(PY_CALL_TRAMPOLINE)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif // ndef Py_EMSCRIPTEN_SIGNAL_H
							 |