 706bfaf5e1
			
		
	
	
		706bfaf5e1
		
	
	
	
	
		
			
			Although constructor methods can improve clarity, there are many
contexts in which it is useful to use range() as a shape: notably
Layout, but also Const and AnyConst/AnyValue. Instead of duplicating
these constructor methods everywhere (which is not even easily
possible for Layout), use casting to Shape, introduced in 6aabdc0a.
Fixes #225.
		
	
			
		
			
				
	
	
		
			65 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			65 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from nmigen import *
 | |
| from nmigen.cli import main
 | |
| 
 | |
| 
 | |
| class UARTReceiver(Elaboratable):
 | |
|     def __init__(self, divisor):
 | |
|         self.divisor = divisor
 | |
| 
 | |
|         self.i    = Signal()
 | |
|         self.data = Signal(8)
 | |
|         self.rdy  = Signal()
 | |
|         self.ack  = Signal()
 | |
|         self.err  = Signal()
 | |
| 
 | |
|     def elaborate(self, platform):
 | |
|         m = Module()
 | |
| 
 | |
|         ctr = Signal(range(self.divisor))
 | |
|         stb = Signal()
 | |
|         with m.If(ctr == 0):
 | |
|             m.d.sync += ctr.eq(self.divisor - 1)
 | |
|             m.d.comb += stb.eq(1)
 | |
|         with m.Else():
 | |
|             m.d.sync += ctr.eq(ctr - 1)
 | |
| 
 | |
|         bit = Signal(3)
 | |
|         with m.FSM() as fsm:
 | |
|             with m.State("START"):
 | |
|                 with m.If(~self.i):
 | |
|                     m.next = "DATA"
 | |
|                     m.d.sync += [
 | |
|                         ctr.eq(self.divisor // 2),
 | |
|                         bit.eq(7),
 | |
|                     ]
 | |
|             with m.State("DATA"):
 | |
|                 with m.If(stb):
 | |
|                     m.d.sync += [
 | |
|                         bit.eq(bit - 1),
 | |
|                         self.data.eq(Cat(self.i, self.data))
 | |
|                     ]
 | |
|                     with m.If(bit == 0):
 | |
|                         m.next = "STOP"
 | |
|             with m.State("STOP"):
 | |
|                 with m.If(stb):
 | |
|                     with m.If(self.i):
 | |
|                         m.next = "DONE"
 | |
|                     with m.Else():
 | |
|                         m.next = "ERROR"
 | |
| 
 | |
|             with m.State("DONE"):
 | |
|                 m.d.comb += self.rdy.eq(1)
 | |
|                 with m.If(self.ack):
 | |
|                     m.next = "START"
 | |
| 
 | |
|             m.d.comb += self.err.eq(fsm.ongoing("ERROR"))
 | |
|             with m.State("ERROR"):
 | |
|                 pass
 | |
| 
 | |
|         return m
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     rx = UARTReceiver(20)
 | |
|     main(rx, ports=[rx.i, rx.data, rx.rdy, rx.ack, rx.err])
 |