| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  | # -*- coding: utf-8 -*- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | Type-transformation rules. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __author__     = "Lluís Vilanova <vilanova@ac.upc.edu>" | 
					
						
							| 
									
										
										
										
											2016-02-25 17:43:38 +01:00
										 |  |  | __copyright__  = "Copyright 2012-2016, Lluís Vilanova <vilanova@ac.upc.edu>" | 
					
						
							| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  | __license__    = "GPL version 2 or (at your option) any later version" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __maintainer__ = "Stefan Hajnoczi" | 
					
						
							| 
									
										
										
										
											2020-05-11 10:28:16 +02:00
										 |  |  | __email__      = "stefanha@redhat.com" | 
					
						
							| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _transform_type(type_, trans): | 
					
						
							|  |  |  |     if isinstance(trans, str): | 
					
						
							|  |  |  |         return trans | 
					
						
							|  |  |  |     elif isinstance(trans, dict): | 
					
						
							|  |  |  |         if type_ in trans: | 
					
						
							|  |  |  |             return _transform_type(type_, trans[type_]) | 
					
						
							|  |  |  |         elif None in trans: | 
					
						
							|  |  |  |             return _transform_type(type_, trans[None]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return type_ | 
					
						
							|  |  |  |     elif callable(trans): | 
					
						
							|  |  |  |         return trans(type_) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise ValueError("Invalid type transformation rule: %s" % trans) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def transform_type(type_, *trans): | 
					
						
							|  |  |  |     """Return a new type transformed according to the given rules.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Applies each of the transformation rules in trans in order. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If an element of trans is a string, return it. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If an element of trans is a function, call it with type_ as its only | 
					
						
							|  |  |  |     argument. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If an element of trans is a dict, search type_ in its keys. If type_ is | 
					
						
							|  |  |  |     a key, use the value as a transformation rule for type_. Otherwise, if | 
					
						
							|  |  |  |     None is a key use the value as a transformation rule for type_. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Otherwise, return type_. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Parameters | 
					
						
							|  |  |  |     ---------- | 
					
						
							|  |  |  |     type_ : str | 
					
						
							|  |  |  |         Type to transform. | 
					
						
							|  |  |  |     trans : list of function or dict | 
					
						
							|  |  |  |         Type transformation rules. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     if len(trans) == 0: | 
					
						
							|  |  |  |         raise ValueError | 
					
						
							|  |  |  |     res = type_ | 
					
						
							|  |  |  |     for t in trans: | 
					
						
							|  |  |  |         res = _transform_type(res, t) | 
					
						
							|  |  |  |     return res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # tcg -> host | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _tcg_2_host(type_): | 
					
						
							|  |  |  |     if type_ == "TCGv": | 
					
						
							|  |  |  |         # force a fixed-size type (target-independent) | 
					
						
							|  |  |  |         return "uint64_t" | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return type_ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TCG_2_HOST = { | 
					
						
							|  |  |  |     "TCGv_i32": "uint32_t", | 
					
						
							|  |  |  |     "TCGv_i64": "uint64_t", | 
					
						
							|  |  |  |     "TCGv_ptr": "void *", | 
					
						
							|  |  |  |     None: _tcg_2_host, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # host -> host compatible with tcg sizes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HOST_2_TCG_COMPAT = { | 
					
						
							|  |  |  |     "uint8_t": "uint32_t", | 
					
						
							| 
									
										
										
										
											2019-06-28 20:54:11 +01:00
										 |  |  |     "uint16_t": "uint32_t", | 
					
						
							| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # host/tcg -> tcg | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _host_2_tcg(type_): | 
					
						
							|  |  |  |     if type_.startswith("TCGv"): | 
					
						
							|  |  |  |         return type_ | 
					
						
							|  |  |  |     raise ValueError("Don't know how to translate '%s' into a TCG type\n" % type_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HOST_2_TCG = { | 
					
						
							|  |  |  |     "uint32_t": "TCGv_i32", | 
					
						
							|  |  |  |     "uint64_t": "TCGv_i64", | 
					
						
							|  |  |  |     "void *"  : "TCGv_ptr", | 
					
						
							| 
									
										
										
										
											2016-02-25 17:43:38 +01:00
										 |  |  |     "CPUArchState *": "TCGv_env", | 
					
						
							| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  |     None: _host_2_tcg, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # tcg -> tcg helper definition | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _tcg_2_helper_def(type_): | 
					
						
							|  |  |  |     if type_ == "TCGv": | 
					
						
							|  |  |  |         return "target_ulong" | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return type_ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TCG_2_TCG_HELPER_DEF = { | 
					
						
							|  |  |  |     "TCGv_i32": "uint32_t", | 
					
						
							|  |  |  |     "TCGv_i64": "uint64_t", | 
					
						
							|  |  |  |     "TCGv_ptr": "void *", | 
					
						
							|  |  |  |     None: _tcg_2_helper_def, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # tcg -> tcg helper declaration | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _tcg_2_tcg_helper_decl_error(type_): | 
					
						
							|  |  |  |     raise ValueError("Don't know how to translate type '%s' into a TCG helper declaration type\n" % type_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TCG_2_TCG_HELPER_DECL = { | 
					
						
							|  |  |  |     "TCGv"    : "tl", | 
					
						
							|  |  |  |     "TCGv_ptr": "ptr", | 
					
						
							|  |  |  |     "TCGv_i32": "i32", | 
					
						
							|  |  |  |     "TCGv_i64": "i64", | 
					
						
							| 
									
										
										
										
											2016-02-25 17:43:38 +01:00
										 |  |  |     "TCGv_env": "env", | 
					
						
							| 
									
										
										
										
											2014-05-30 14:11:32 +02:00
										 |  |  |     None: _tcg_2_tcg_helper_decl_error, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # host/tcg -> tcg temporal constant allocation | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _host_2_tcg_tmp_new(type_): | 
					
						
							|  |  |  |     if type_.startswith("TCGv"): | 
					
						
							|  |  |  |         return "tcg_temp_new_nop" | 
					
						
							|  |  |  |     raise ValueError("Don't know how to translate type '%s' into a TCG temporal allocation" % type_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HOST_2_TCG_TMP_NEW = { | 
					
						
							|  |  |  |     "uint32_t": "tcg_const_i32", | 
					
						
							|  |  |  |     "uint64_t": "tcg_const_i64", | 
					
						
							|  |  |  |     "void *"  : "tcg_const_ptr", | 
					
						
							|  |  |  |     None: _host_2_tcg_tmp_new, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################## | 
					
						
							|  |  |  | # host/tcg -> tcg temporal constant deallocation | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _host_2_tcg_tmp_free(type_): | 
					
						
							|  |  |  |     if type_.startswith("TCGv"): | 
					
						
							|  |  |  |         return "tcg_temp_free_nop" | 
					
						
							|  |  |  |     raise ValueError("Don't know how to translate type '%s' into a TCG temporal deallocation" % type_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HOST_2_TCG_TMP_FREE = { | 
					
						
							|  |  |  |     "uint32_t": "tcg_temp_free_i32", | 
					
						
							|  |  |  |     "uint64_t": "tcg_temp_free_i64", | 
					
						
							|  |  |  |     "void *"  : "tcg_temp_free_ptr", | 
					
						
							|  |  |  |     None: _host_2_tcg_tmp_free, | 
					
						
							|  |  |  |     } |