fhdl.ast, back.pysim: implement shifts.
This commit is contained in:
		
							parent
							
								
									46f5addf05
								
							
						
					
					
						commit
						db4600d52b
					
				|  | @ -84,6 +84,14 @@ class _RHSValueCompiler(ValueTransformer): | |||
|                 return lambda state: normalize(lhs(state) |  rhs(state), shape) | ||||
|             if value.op == "^": | ||||
|                 return lambda state: normalize(lhs(state) ^  rhs(state), shape) | ||||
|             if value.op == "<<": | ||||
|                 def sshl(lhs, rhs): | ||||
|                     return lhs << rhs if rhs >= 0 else lhs >> -rhs | ||||
|                 return lambda state: normalize(sshl(lhs(state), rhs(state)), shape) | ||||
|             if value.op == ">>": | ||||
|                 def sshr(lhs, rhs): | ||||
|                     return lhs >> rhs if rhs >= 0 else lhs << -rhs | ||||
|                 return lambda state: normalize(sshr(lhs(state), rhs(state)), shape) | ||||
|             if value.op == "==": | ||||
|                 return lambda state: normalize(lhs(state) == rhs(state), shape) | ||||
|             if value.op == "!=": | ||||
|  |  | |||
|  | @ -206,8 +206,8 @@ class _ValueTransformer(xfrm.ValueTransformer): | |||
|         (2, "/"):    "$div", | ||||
|         (2, "%"):    "$mod", | ||||
|         (2, "**"):   "$pow", | ||||
|         (2, "<<<"):  "$sshl", | ||||
|         (2, ">>>"):  "$sshr", | ||||
|         (2, "<<"):   "$sshl", | ||||
|         (2, ">>"):   "$sshr", | ||||
|         (2, "&"):    "$and", | ||||
|         (2, "^"):    "$xor", | ||||
|         (2, "|"):    "$or", | ||||
|  |  | |||
|  | @ -75,13 +75,13 @@ class Value(metaclass=ABCMeta): | |||
|     def __rdiv__(self, other): | ||||
|         return Operator("/", [other, self]) | ||||
|     def __lshift__(self, other): | ||||
|         return Operator("<<<", [self, other]) | ||||
|         return Operator("<<", [self, other]) | ||||
|     def __rlshift__(self, other): | ||||
|         return Operator("<<<", [other, self]) | ||||
|         return Operator("<<", [other, self]) | ||||
|     def __rshift__(self, other): | ||||
|         return Operator(">>>", [self, other]) | ||||
|         return Operator(">>", [self, other]) | ||||
|     def __rrshift__(self, other): | ||||
|         return Operator(">>>", [other, self]) | ||||
|         return Operator(">>", [other, self]) | ||||
|     def __and__(self, other): | ||||
|         return Operator("&", [self, other]) | ||||
|     def __rand__(self, other): | ||||
|  | @ -306,13 +306,13 @@ class Operator(Value): | |||
|                 return 1, False | ||||
|             if self.op in ("&", "^", "|"): | ||||
|                 return self._bitwise_binary_shape(*op_shapes) | ||||
|             if self.op == "<<<": | ||||
|             if self.op == "<<": | ||||
|                 if b_sign: | ||||
|                     extra = 2 ** (b_bits - 1) - 1 | ||||
|                 else: | ||||
|                     extra = 2**b_bits - 1 | ||||
|                     extra = 2 ** (b_bits)     - 1 | ||||
|                 return a_bits + extra, a_sign | ||||
|             if self.op == ">>>": | ||||
|             if self.op == ">>": | ||||
|                 if b_sign: | ||||
|                     extra = 2 ** (b_bits - 1) | ||||
|                 else: | ||||
|  |  | |||
|  | @ -182,6 +182,20 @@ class OperatorTestCase(FHDLTestCase): | |||
|         v5 = 10 ^ Const(0, 4) | ||||
|         self.assertEqual(v5.shape(), (4, False)) | ||||
| 
 | ||||
|     def test_shl(self): | ||||
|         v1 = Const(1, 4) << Const(4) | ||||
|         self.assertEqual(repr(v1), "(<< (const 4'd1) (const 3'd4))") | ||||
|         self.assertEqual(v1.shape(), (11, False)) | ||||
|         v2 = Const(1, 4) << Const(-3) | ||||
|         self.assertEqual(v2.shape(), (7, False)) | ||||
| 
 | ||||
|     def test_shr(self): | ||||
|         v1 = Const(1, 4) >> Const(4) | ||||
|         self.assertEqual(repr(v1), "(>> (const 4'd1) (const 3'd4))") | ||||
|         self.assertEqual(v1.shape(), (4, False)) | ||||
|         v2 = Const(1, 4) >> Const(-3) | ||||
|         self.assertEqual(v2.shape(), (8, False)) | ||||
| 
 | ||||
|     def test_lt(self): | ||||
|         v = Const(0, 4) < Const(0, 6) | ||||
|         self.assertEqual(repr(v), "(< (const 4'd0) (const 6'd0))") | ||||
|  |  | |||
|  | @ -71,6 +71,18 @@ class SimulatorUnitTestCase(FHDLTestCase): | |||
|         stmt = lambda a, b: a ^ b | ||||
|         self.assertOperator(stmt, [C(0b1100, 4), C(0b1010, 4)], C(0b0110, 4)) | ||||
| 
 | ||||
|     def test_shl(self): | ||||
|         stmt = lambda a, b: a << b | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(0)],  C(0b1001,    5)) | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(3)],  C(0b1001000, 7)) | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(-2)], C(0b10,      7)) | ||||
| 
 | ||||
|     def test_shr(self): | ||||
|         stmt = lambda a, b: a >> b | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(0)],  C(0b1001,    4)) | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(2)],  C(0b10,      4)) | ||||
|         self.assertOperator(stmt, [C(0b1001, 4), C(-2)], C(0b100100,  5)) | ||||
| 
 | ||||
|     def test_eq(self): | ||||
|         stmt = lambda a, b: a == b | ||||
|         self.assertOperator(stmt, [C(0, 4), C(0, 4)], C(1)) | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue
	
	 whitequark
						whitequark