litex/misoclib/mem/litesata/common.py

276 lines
6.6 KiB
Python
Raw Normal View History

2015-01-06 10:48:19 -05:00
import math
2014-11-03 12:54:41 -05:00
from migen.fhdl.std import *
from migen.fhdl.decorators import ModuleTransformer
from migen.genlib.resetsync import *
from migen.genlib.fsm import *
from migen.genlib.record import *
from migen.genlib.misc import chooser, optree, Counter, WaitTimer
2015-01-16 17:52:41 -05:00
from migen.genlib.cdc import *
2014-12-03 10:32:55 -05:00
from migen.flow.actor import *
2015-01-16 17:52:41 -05:00
from migen.flow.plumbing import Multiplexer, Demultiplexer
from migen.actorlib.fifo import *
2015-01-16 17:52:41 -05:00
from migen.actorlib.structuring import Pipeline, Converter
from migen.actorlib.packet import Buffer
from migen.actorlib.misc import BufferizeEndpoints
from migen.actorlib.packet import HeaderField, Header
2014-11-03 12:54:41 -05:00
2015-01-22 11:15:12 -05:00
bitrates = {
2015-04-13 09:25:40 -04:00
"sata_gen3": 6.0,
"sata_gen2": 3.0,
"sata_gen1": 1.5,
2015-01-22 11:15:12 -05:00
}
2015-01-19 11:52:05 -05:00
frequencies = {
2015-04-13 09:25:40 -04:00
"sata_gen3": 150.0,
"sata_gen2": 75.0,
"sata_gen1": 37.5,
2015-01-19 11:52:05 -05:00
}
2015-01-22 11:15:12 -05:00
# PHY / Link Layers
primitives = {
2015-04-13 09:25:40 -04:00
"ALIGN": 0x7B4A4ABC,
"CONT": 0X9999AA7C,
"SYNC": 0xB5B5957C,
"R_RDY": 0x4A4A957C,
"R_OK": 0x3535B57C,
"R_ERR": 0x5656B57C,
"R_IP": 0X5555B57C,
"X_RDY": 0x5757B57C,
"CONT": 0x9999AA7C,
"WTRM": 0x5858B57C,
"SOF": 0x3737B57C,
"EOF": 0xD5D5B57C,
"HOLD": 0xD5D5AA7C,
"HOLDA": 0X9595AA7C
}
2015-04-13 09:12:39 -04:00
2014-12-03 05:12:26 -05:00
def is_primitive(dword):
for k, v in primitives.items():
if dword == v:
return True
return False
2014-12-03 05:12:26 -05:00
2015-04-13 09:12:39 -04:00
2014-12-03 05:12:26 -05:00
def decode_primitive(dword):
for k, v in primitives.items():
if dword == v:
return k
return ""
2014-12-03 05:12:26 -05:00
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def phy_description(dw):
layout = [
("data", dw),
("charisk", dw//8),
]
return EndpointDescription(layout, packetized=False)
2014-11-04 11:35:46 -05:00
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def link_description(dw):
layout = [
("d", dw),
("error", 1)
]
return EndpointDescription(layout, packetized=True)
2014-12-03 10:32:55 -05:00
# Transport Layer
2015-01-06 10:48:19 -05:00
fis_max_dwords = 2048
fis_types = {
"REG_H2D": 0x27,
"REG_D2H": 0x34,
"DMA_ACTIVATE_D2H": 0x39,
"PIO_SETUP_D2H": 0x5F,
"DATA": 0x46
}
fis_reg_h2d_header_length = 5
fis_reg_h2d_header_fields = {
"type": HeaderField(0*4, 0, 8),
"pm_port": HeaderField(0*4, 8, 4),
"c": HeaderField(0*4, 15, 1),
"command": HeaderField(0*4, 16, 8),
"features_lsb": HeaderField(0*4, 24, 8),
2015-04-13 09:12:39 -04:00
"lba_lsb": HeaderField(1*4, 0, 24),
"device": HeaderField(1*4, 24, 8),
"lba_msb": HeaderField(2*4, 0, 24),
"features_msb": HeaderField(2*4, 24, 8),
"count": HeaderField(3*4, 0, 16),
"icc": HeaderField(3*4, 16, 8),
"control": HeaderField(3*4, 24, 8)
}
fis_reg_h2d_header = Header(fis_reg_h2d_header_fields,
fis_reg_h2d_header_length,
swap_field_bytes=False)
fis_reg_d2h_header_length = 5
fis_reg_d2h_header_fields = {
"type": HeaderField(0*4, 0, 8),
"pm_port": HeaderField(0*4, 8, 4),
"i": HeaderField(0*4, 14, 1),
"status": HeaderField(0*4, 16, 8),
"error": HeaderField(0*4, 24, 8),
"lba_lsb": HeaderField(1*4, 0, 24),
"device": HeaderField(1*4, 24, 8),
"lba_msb": HeaderField(2*4, 0, 24),
"count": HeaderField(3*4, 0, 16)
}
fis_reg_d2h_header = Header(fis_reg_d2h_header_fields,
fis_reg_d2h_header_length,
swap_field_bytes=False)
fis_dma_activate_d2h_header_length = 1
fis_dma_activate_d2h_header_fields = {
"type": HeaderField(0*4, 0, 8),
"pm_port": HeaderField(0*4, 8, 4)
}
fis_dma_activate_d2h_header = Header(fis_dma_activate_d2h_header_fields,
fis_dma_activate_d2h_header_length,
swap_field_bytes=False)
fis_pio_setup_d2h_header_length = 5
fis_pio_setup_d2h_header_fields = {
"type": HeaderField(0*4, 0, 8),
"pm_port": HeaderField(0*4, 8, 4),
"d": HeaderField(0*4, 13, 1),
"i": HeaderField(0*4, 14, 1),
"status": HeaderField(0*4, 16, 8),
"error": HeaderField(0*4, 24, 8),
2015-01-16 16:49:34 -05:00
"lba_lsb": HeaderField(1*4, 0, 24),
2015-01-16 16:49:34 -05:00
"lba_msb": HeaderField(2*4, 0, 24),
2015-01-16 16:49:34 -05:00
"count": HeaderField(3*4, 0, 16),
2015-01-16 16:49:34 -05:00
"transfer_count": HeaderField(4*4, 0, 16),
2015-01-16 16:49:34 -05:00
}
fis_pio_setup_d2h_header = Header(fis_pio_setup_d2h_header_fields,
fis_pio_setup_d2h_header_length,
swap_field_bytes=False)
2015-01-16 16:49:34 -05:00
fis_data_header_length = 1
fis_data_header_fields = {
"type": HeaderField(0, 0, 8)
}
fis_data_header = Header(fis_data_header_fields,
fis_data_header_length,
swap_field_bytes=False)
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def transport_tx_description(dw):
layout = [
("type", 8),
("pm_port", 4),
("c", 1),
("command", 8),
("features", 16),
("lba", 48),
("device", 8),
("count", 16),
("icc", 8),
("control", 8),
("data", dw)
]
return EndpointDescription(layout, packetized=True)
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def transport_rx_description(dw):
layout = [
("type", 8),
("pm_port", 4),
("r", 1),
("d", 1),
("i", 1),
("status", 8),
("errors", 8),
("lba", 48),
("device", 8),
("count", 16),
("transfer_count", 16),
("data", dw),
("error", 1)
]
return EndpointDescription(layout, packetized=True)
# Command Layer
regs = {
2015-04-13 09:25:40 -04:00
"WRITE_DMA_EXT": 0x35,
"READ_DMA_EXT": 0x25,
"IDENTIFY_DEVICE": 0xEC
}
2015-01-20 11:14:01 -05:00
reg_d2h_status = {
2015-04-13 09:25:40 -04:00
"bsy": 7,
"drdy": 6,
"df": 5,
"se": 5,
"dwe": 4,
"drq": 3,
"ae": 2,
"sns": 1,
"cc": 0,
"err": 0
2015-01-20 11:14:01 -05:00
}
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def command_tx_description(dw):
layout = [
("write", 1),
("read", 1),
("identify", 1),
("sector", 48),
("count", 16),
("data", dw)
]
return EndpointDescription(layout, packetized=True)
2015-04-13 09:12:39 -04:00
2014-12-14 04:52:56 -05:00
def command_rx_description(dw):
layout = [
("write", 1),
("read", 1),
("identify", 1),
("last", 1),
("failed", 1),
("data", dw)
]
return EndpointDescription(layout, packetized=True)
2015-04-13 09:12:39 -04:00
def command_rx_cmd_description(dw):
layout = [
("write", 1),
("read", 1),
("identify", 1),
("last", 1),
("failed", 1)
]
return EndpointDescription(layout, packetized=False)
2015-04-13 09:12:39 -04:00
def command_rx_data_description(dw):
layout = [
("data", dw)
]
return EndpointDescription(layout, packetized=True)
2015-01-06 10:48:19 -05:00
# HDD
2015-04-13 09:51:17 -04:00
logical_sector_size = 512 # constant since all HDDs use this
2015-01-06 10:48:19 -05:00
2015-04-13 09:12:39 -04:00
2015-01-06 10:48:19 -05:00
def dwords2sectors(n):
return math.ceil(n*4/logical_sector_size)
2015-01-06 10:48:19 -05:00
2015-04-13 09:12:39 -04:00
2015-01-06 10:48:19 -05:00
def sectors2dwords(n):
return n*logical_sector_size//4