| 
									
										
										
										
											2010-03-11 23:39:40 +00:00
										 |  |  | #! /usr/bin/env python | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  | """Generate C code for the jump table of the threaded code interpreter
 | 
					
						
							|  |  |  | (for compilers supporting computed gotos or "labels-as-values", such as gcc). | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import os | 
					
						
							| 
									
										
										
										
											2016-03-25 11:54:47 +01:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-28 16:17:58 -07:00
										 |  |  | # 2023-04-27(warsaw): Pre-Python 3.12, this would catch ImportErrors and try to | 
					
						
							|  |  |  | # import imp, and then use imp.load_module().  The imp module was removed in | 
					
						
							|  |  |  | # Python 3.12 (and long deprecated before that), and it's unclear under what | 
					
						
							|  |  |  | # conditions this import will now fail, so the fallback was simply removed. | 
					
						
							|  |  |  | from importlib.machinery import SourceFileLoader | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def find_module(modname): | 
					
						
							|  |  |  |     """Finds and returns a module in the local dist/checkout.
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     modpath = os.path.join( | 
					
						
							|  |  |  |         os.path.dirname(os.path.dirname(__file__)), "Lib", modname + ".py") | 
					
						
							|  |  |  |     return SourceFileLoader(modname, modpath).load_module() | 
					
						
							| 
									
										
										
										
											2016-03-26 01:04:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  | def write_contents(f): | 
					
						
							|  |  |  |     """Write C code contents to the target file object.
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2016-03-26 01:04:37 +01:00
										 |  |  |     opcode = find_module('opcode') | 
					
						
							| 
									
										
										
										
											2023-06-19 23:47:04 +01:00
										 |  |  |     _opcode_metadata = find_module('_opcode_metadata') | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  |     targets = ['_unknown_opcode'] * 256 | 
					
						
							|  |  |  |     for opname, op in opcode.opmap.items(): | 
					
						
							| 
									
										
										
										
											2022-07-01 15:33:35 +01:00
										 |  |  |         if not opcode.is_pseudo(op): | 
					
						
							|  |  |  |             targets[op] = "TARGET_%s" % opname | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     next_op = 1 | 
					
						
							| 
									
										
										
										
											2023-06-19 23:47:04 +01:00
										 |  |  |     for opname in _opcode_metadata._specialized_instructions: | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         while targets[next_op] != '_unknown_opcode': | 
					
						
							|  |  |  |             next_op += 1 | 
					
						
							|  |  |  |         targets[next_op] = "TARGET_%s" % opname | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  |     f.write("static void *opcode_targets[256] = {\n") | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     f.write(",\n".join(["    &&%s" % s for s in targets])) | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  |     f.write("\n};\n") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 11:54:47 +01:00
										 |  |  | def main(): | 
					
						
							|  |  |  |     if len(sys.argv) >= 3: | 
					
						
							|  |  |  |         sys.exit("Too many arguments") | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  |     if len(sys.argv) == 2: | 
					
						
							|  |  |  |         target = sys.argv[1] | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         target = "Python/opcode_targets.h" | 
					
						
							| 
									
										
										
										
											2016-03-25 11:54:47 +01:00
										 |  |  |     with open(target, "w") as f: | 
					
						
							| 
									
										
										
										
											2009-01-25 16:34:23 +00:00
										 |  |  |         write_contents(f) | 
					
						
							| 
									
										
										
										
											2016-03-25 11:54:47 +01:00
										 |  |  |     print("Jump table written into %s" % target) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     main() |