| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | #include <Python.h>
 | 
					
						
							| 
									
										
										
										
											2017-05-22 19:46:40 -07:00
										 |  |  | #include <inttypes.h>
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | /*********************************************************
 | 
					
						
							|  |  |  |  * Embedded interpreter tests that need a custom exe | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Executed via 'EmbeddingTests' in Lib/test/test_capi.py | 
					
						
							|  |  |  |  *********************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void _testembed_Py_Initialize(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* HACK: the "./" at front avoids a search along the PATH in
 | 
					
						
							|  |  |  |        Modules/getpath.c */ | 
					
						
							|  |  |  |     Py_SetProgramName(L"./_testembed"); | 
					
						
							|  |  |  |     Py_Initialize(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*****************************************************
 | 
					
						
							| 
									
										
										
										
											2016-04-19 04:03:41 +00:00
										 |  |  |  * Test repeated initialisation and subinterpreters | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  |  *****************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void print_subinterp(void) | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-05-22 19:46:40 -07:00
										 |  |  |     /* Output information about the interpreter in the format
 | 
					
						
							|  |  |  |        expected in Lib/test/test_capi.py (test_subinterps). */ | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |     PyThreadState *ts = PyThreadState_Get(); | 
					
						
							| 
									
										
										
										
											2017-05-22 19:46:40 -07:00
										 |  |  |     PyInterpreterState *interp = ts->interp; | 
					
						
							|  |  |  |     int64_t id = PyInterpreterState_GetID(interp); | 
					
						
							|  |  |  |     printf("interp %lu <0x%" PRIXPTR ">, thread state <0x%" PRIXPTR ">: ", | 
					
						
							|  |  |  |             id, (uintptr_t)interp, (uintptr_t)ts); | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |     fflush(stdout); | 
					
						
							|  |  |  |     PyRun_SimpleString( | 
					
						
							|  |  |  |         "import sys;" | 
					
						
							|  |  |  |         "print('id(modules) =', id(sys.modules));" | 
					
						
							|  |  |  |         "sys.stdout.flush()" | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  | static int test_repeated_init_and_subinterpreters(void) | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     PyThreadState *mainstate, *substate; | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #ifdef WITH_THREAD
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |     PyGILState_STATE gilstate; | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |     int i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-16 00:17:42 -04:00
										 |  |  |     for (i=0; i<15; i++) { | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |         printf("--- Pass %d ---\n", i); | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  |         _testembed_Py_Initialize(); | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |         mainstate = PyThreadState_Get(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #ifdef WITH_THREAD
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |         PyEval_InitThreads(); | 
					
						
							|  |  |  |         PyEval_ReleaseThread(mainstate); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         gilstate = PyGILState_Ensure(); | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |         print_subinterp(); | 
					
						
							|  |  |  |         PyThreadState_Swap(NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (j=0; j<3; j++) { | 
					
						
							|  |  |  |             substate = Py_NewInterpreter(); | 
					
						
							|  |  |  |             print_subinterp(); | 
					
						
							|  |  |  |             Py_EndInterpreter(substate); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         PyThreadState_Swap(mainstate); | 
					
						
							|  |  |  |         print_subinterp(); | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #ifdef WITH_THREAD
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  |         PyGILState_Release(gilstate); | 
					
						
							| 
									
										
										
										
											2011-04-26 23:37:02 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         PyEval_RestoreThread(mainstate); | 
					
						
							|  |  |  |         Py_Finalize(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*****************************************************
 | 
					
						
							|  |  |  |  * Test forcing a particular IO encoding | 
					
						
							|  |  |  |  *****************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void check_stdio_details(const char *encoding, const char * errors) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Output info for the test case to check */ | 
					
						
							|  |  |  |     if (encoding) { | 
					
						
							|  |  |  |         printf("Expected encoding: %s\n", encoding); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         printf("Expected encoding: default\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (errors) { | 
					
						
							|  |  |  |         printf("Expected errors: %s\n", errors); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         printf("Expected errors: default\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  |     /* Force the given IO encoding */ | 
					
						
							|  |  |  |     Py_SetStandardStreamEncoding(encoding, errors); | 
					
						
							|  |  |  |     _testembed_Py_Initialize(); | 
					
						
							|  |  |  |     PyRun_SimpleString( | 
					
						
							|  |  |  |         "import sys;" | 
					
						
							|  |  |  |         "print('stdin: {0.encoding}:{0.errors}'.format(sys.stdin));" | 
					
						
							|  |  |  |         "print('stdout: {0.encoding}:{0.errors}'.format(sys.stdout));" | 
					
						
							|  |  |  |         "print('stderr: {0.encoding}:{0.errors}'.format(sys.stderr));" | 
					
						
							|  |  |  |         "sys.stdout.flush()" | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     Py_Finalize(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  | static int test_forced_io_encoding(void) | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | { | 
					
						
							|  |  |  |     /* Check various combinations */ | 
					
						
							|  |  |  |     printf("--- Use defaults ---\n"); | 
					
						
							|  |  |  |     check_stdio_details(NULL, NULL); | 
					
						
							|  |  |  |     printf("--- Set errors only ---\n"); | 
					
						
							| 
									
										
										
										
											2014-03-18 02:38:12 +01:00
										 |  |  |     check_stdio_details(NULL, "ignore"); | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  |     printf("--- Set encoding only ---\n"); | 
					
						
							|  |  |  |     check_stdio_details("latin-1", NULL); | 
					
						
							|  |  |  |     printf("--- Set encoding and errors ---\n"); | 
					
						
							| 
									
										
										
										
											2014-03-18 02:38:12 +01:00
										 |  |  |     check_stdio_details("latin-1", "replace"); | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Check calling after initialization fails */ | 
					
						
							|  |  |  |     Py_Initialize(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (Py_SetStandardStreamEncoding(NULL, NULL) == 0) { | 
					
						
							|  |  |  |         printf("Unexpected success calling Py_SetStandardStreamEncoding"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_Finalize(); | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  | /* *********************************************************
 | 
					
						
							|  |  |  |  * List of test cases and the function that implements it. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Names are compared case-sensitively with the first | 
					
						
							|  |  |  |  * argument. If no match is found, or no first argument was | 
					
						
							|  |  |  |  * provided, the names of all test cases are printed and | 
					
						
							|  |  |  |  * the exit code will be -1. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The int returned from test functions is used as the exit | 
					
						
							|  |  |  |  * code, and test_capi treats all non-zero exit codes as a | 
					
						
							|  |  |  |  * failed test.  | 
					
						
							|  |  |  |  *********************************************************/ | 
					
						
							|  |  |  | struct TestCase | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  |     const char *name; | 
					
						
							|  |  |  |     int (*func)(void); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct TestCase TestCases[] = { | 
					
						
							|  |  |  |     { "forced_io_encoding", test_forced_io_encoding }, | 
					
						
							|  |  |  |     { "repeated_init_and_subinterpreters", test_repeated_init_and_subinterpreters }, | 
					
						
							|  |  |  |     { NULL, NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  | int main(int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  |     if (argc > 1) { | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  |         for (struct TestCase *tc = TestCases; tc && tc->name; tc++) { | 
					
						
							|  |  |  |             if (strcmp(argv[1], tc->name) == 0) | 
					
						
							|  |  |  |                 return (*tc->func)(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-10-17 22:35:35 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-01 20:25:03 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* No match found, or no test name provided, so display usage */ | 
					
						
							|  |  |  |     printf("Python " PY_VERSION " _testembed executable for embedded interpreter tests\n" | 
					
						
							|  |  |  |            "Normally executed via 'EmbeddingTests' in Lib/test/test_capi.py\n\n" | 
					
						
							|  |  |  |            "Usage: %s TESTNAME\n\nAll available tests:\n", argv[0]); | 
					
						
							|  |  |  |     for (struct TestCase *tc = TestCases; tc && tc->name; tc++) { | 
					
						
							|  |  |  |         printf("  %s\n", tc->name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Non-zero exit code will cause test_capi.py tests to fail.
 | 
					
						
							|  |  |  |        This is intentional. */ | 
					
						
							|  |  |  |     return -1; | 
					
						
							| 
									
										
										
										
											2011-04-25 21:21:07 +02:00
										 |  |  | } |