litex/milkymist/dvisampler/edid.py

178 lines
4.2 KiB
Python
Raw Normal View History

2013-05-22 11:10:13 -04:00
from migen.fhdl.std import *
from migen.fhdl.specials import Tristate
2013-03-13 14:56:26 -04:00
from migen.genlib.cdc import MultiReg
2013-06-25 16:25:10 -04:00
from migen.genlib.fsm import FSM, NextState
2013-03-13 14:56:26 -04:00
from migen.genlib.misc import chooser
2013-03-30 12:28:15 -04:00
from migen.bank.description import AutoCSR
2013-03-13 14:56:26 -04:00
_default_edid = [
2013-03-23 08:48:40 -04:00
0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x3D, 0x17, 0x32, 0x12, 0x2A, 0x6A, 0xBF, 0x00,
0x05, 0x17, 0x01, 0x03, 0x80, 0x28, 0x1E, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x2E, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01,
0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0xE2, 0x0E, 0x20, 0x20, 0x31, 0x58, 0x13, 0x20, 0x20, 0x80,
0x14, 0x00, 0x28, 0x1E, 0x00, 0x00, 0x00, 0x1E, 0x00, 0x00, 0x00, 0xFC, 0x00, 0x4D, 0x31, 0x20,
0x44, 0x56, 0x49, 0x20, 0x6D, 0x69, 0x78, 0x65, 0x72, 0x0A, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF,
2013-03-13 14:56:26 -04:00
]
2013-03-30 12:28:15 -04:00
class EDID(Module, AutoCSR):
2013-03-26 12:57:17 -04:00
def __init__(self, pads, default=_default_edid):
2013-03-13 14:56:26 -04:00
self.specials.mem = Memory(8, 128, init=default)
###
scl_raw = Signal()
2013-03-13 14:56:26 -04:00
sda_i = Signal()
sda_drv = Signal()
_sda_drv_reg = Signal()
_sda_i_async = Signal()
self.sync += _sda_drv_reg.eq(sda_drv)
self.specials += [
MultiReg(pads.scl, scl_raw),
2013-03-26 12:57:17 -04:00
Tristate(pads.sda, 0, _sda_drv_reg, _sda_i_async),
MultiReg(_sda_i_async, sda_i)
2013-03-13 14:56:26 -04:00
]
scl_i = Signal()
samp_count = Signal(6)
samp_carry = Signal()
self.sync += [
Cat(samp_count, samp_carry).eq(samp_count + 1),
If(samp_carry, scl_i.eq(scl_raw))
]
2013-03-13 14:56:26 -04:00
scl_r = Signal()
sda_r = Signal()
scl_rising = Signal()
sda_rising = Signal()
sda_falling = Signal()
self.sync += [
scl_r.eq(scl_i),
sda_r.eq(sda_i)
]
self.comb += [
scl_rising.eq(scl_i & ~scl_r),
sda_rising.eq(sda_i & ~sda_r),
sda_falling.eq(~sda_i & sda_r)
]
start = Signal()
self.comb += start.eq(scl_i & sda_falling)
din = Signal(8)
counter = Signal(max=9)
self.sync += [
If(start, counter.eq(0)),
If(scl_rising,
If(counter == 8,
counter.eq(0)
).Else(
counter.eq(counter + 1),
din.eq(Cat(sda_i, din[:7]))
)
)
]
is_read = Signal()
update_is_read = Signal()
self.sync += If(update_is_read, is_read.eq(din[0]))
offset_counter = Signal(max=128)
oc_load = Signal()
oc_inc = Signal()
self.sync += [
If(oc_load,
offset_counter.eq(din)
).Elif(oc_inc,
offset_counter.eq(offset_counter + 1)
)
]
rdport = self.mem.get_port()
2013-05-28 09:56:14 -04:00
self.specials += rdport
2013-03-13 14:56:26 -04:00
self.comb += rdport.adr.eq(offset_counter)
data_bit = Signal()
zero_drv = Signal()
data_drv = Signal()
self.comb += If(zero_drv, sda_drv.eq(1)).Elif(data_drv, sda_drv.eq(~data_bit))
data_drv_en = Signal()
data_drv_stop = Signal()
self.sync += If(data_drv_en, data_drv.eq(1)).Elif(data_drv_stop, data_drv.eq(0))
self.sync += If(data_drv_en, chooser(rdport.dat_r, counter, data_bit, 8, reverse=True))
2013-06-25 16:25:10 -04:00
fsm = FSM()
2013-03-13 14:56:26 -04:00
self.submodules += fsm
2013-06-25 16:25:10 -04:00
fsm.act("WAIT_START")
fsm.act("RCV_ADDRESS",
2013-03-13 14:56:26 -04:00
If(counter == 8,
If(din[1:] == 0x50,
update_is_read.eq(1),
2013-06-25 16:25:10 -04:00
NextState("ACK_ADDRESS0")
2013-03-13 14:56:26 -04:00
).Else(
2013-06-25 16:25:10 -04:00
NextState("WAIT_START")
2013-03-13 14:56:26 -04:00
)
)
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_ADDRESS0",
If(~scl_i, NextState("ACK_ADDRESS1"))
2013-03-13 14:56:26 -04:00
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_ADDRESS1",
2013-03-13 14:56:26 -04:00
zero_drv.eq(1),
2013-06-25 16:25:10 -04:00
If(scl_i, NextState("ACK_ADDRESS2"))
2013-03-13 14:56:26 -04:00
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_ADDRESS2",
2013-03-13 14:56:26 -04:00
zero_drv.eq(1),
If(~scl_i,
If(is_read,
2013-06-25 16:25:10 -04:00
NextState("READ")
2013-03-13 14:56:26 -04:00
).Else(
2013-06-25 16:25:10 -04:00
NextState("RCV_OFFSET")
2013-03-13 14:56:26 -04:00
)
)
)
2013-06-25 16:25:10 -04:00
fsm.act("RCV_OFFSET",
2013-03-13 14:56:26 -04:00
If(counter == 8,
oc_load.eq(1),
2013-06-25 16:25:10 -04:00
NextState("ACK_OFFSET0")
2013-03-13 14:56:26 -04:00
)
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_OFFSET0",
If(~scl_i, NextState("ACK_OFFSET1"))
2013-03-13 14:56:26 -04:00
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_OFFSET1",
2013-03-13 14:56:26 -04:00
zero_drv.eq(1),
2013-06-25 16:25:10 -04:00
If(scl_i, NextState("ACK_OFFSET2"))
2013-03-13 14:56:26 -04:00
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_OFFSET2",
2013-03-13 14:56:26 -04:00
zero_drv.eq(1),
2013-06-25 16:25:10 -04:00
If(~scl_i, NextState("RCV_ADDRESS"))
2013-03-13 14:56:26 -04:00
)
2013-06-25 16:25:10 -04:00
fsm.act("READ",
2013-03-13 14:56:26 -04:00
If(~scl_i,
If(counter == 8,
data_drv_stop.eq(1),
2013-06-25 16:25:10 -04:00
NextState("ACK_READ")
2013-03-13 14:56:26 -04:00
).Else(
data_drv_en.eq(1)
)
)
)
2013-06-25 16:25:10 -04:00
fsm.act("ACK_READ",
2013-03-13 14:56:26 -04:00
If(scl_rising,
oc_inc.eq(1),
If(sda_i,
2013-06-25 16:25:10 -04:00
NextState("WAIT_START")
2013-03-13 14:56:26 -04:00
).Else(
2013-06-25 16:25:10 -04:00
NextState("READ")
2013-03-13 14:56:26 -04:00
)
)
)
2013-06-25 16:25:10 -04:00
for state in fsm.actions.keys():
fsm.act(state, If(start, NextState("RCV_ADDRESS")))