草庐IT

FPGA SATA IP控制器的SATA接口调试记录

缥缈孤鸿_jason 2023-04-07 原文

本文档是基于FPGA K7 SATA IP控制器的SATA接口调试记录,接口遵循标准的ACHI协议。

操作系统内核版本:5.4.18

由于K7PCIE只有3个bar,AHCI协议规定SATA控制器是在第四个BAR上,另外由于PCIE配置空间设备类寄存器和能力寄存器未配置成sata设备,导致系统自带的驱动不能没生效。因此需要修改系统sata驱动并重新编译安装。

Sata驱动在内核目录/drivers/ata/下,对于ahci sata对于的驱动文件是ahci.c,单独编译ahci文件需要对该目录下的Makefile文件进行如下修改:

SPDX-License-Identifier: GPL-2.0

#将ahci.c文件编译成模块

CONFIG_SATA_AHCI := m

#CONFIG_PATA_PLATFORM := m

#CONFIG_PATA_OF_PLATFORM := m

#CONFIG_SATA_SIL24 := m

KERNEL_VERSION := $(shell uname -r)

KERNELDIR = /lib/modules/$(KERNEL_VERSION)/build

#INSTDIR = /lib/modules/$(KERNEL_VERSION)/kernel/drivers/ata

KERNEL := kernel-$(KERNEL_VERSION)

PWD := $(shell pwd)

EXTRA_CFLAGS += -DATA_DEBUG

#obj-$(CONFIG_ATA)                 += libata.o

# non-SFF interface

#obj-$(CONFIG_SATA_AHCI)             += ahci.o libahci.o

#编译生成的模块名称为ahci_test.ko

obj-$(CONFIG_SATA_AHCI)               := ahci_test.o

#模块有ahci.c文件编译生成

ahci_test-objs := ahci.o

#obj-$(CONFIG_SATA_ACARD_AHCI)      += acard-ahci.o libahci.o

#obj-$(CONFIG_SATA_AHCI_SEATTLE)    += ahci_seattle.o libahci.o libahci_platform.o

#obj-$(CONFIG_SATA_AHCI_PLATFORM) += ahci_platform.o libahci.o libahci_platform.o

#obj-$(CONFIG_SATA_FSL)                += sata_fsl.o

#obj-$(CONFIG_SATA_GEMINI)       += sata_gemini.o

#obj-$(CONFIG_SATA_INIC162X)     += sata_inic162x.o

#obj-$(CONFIG_SATA_SIL24)   += sata_sil24.o

#obj-$(CONFIG_SATA_SIL24)   += sata_sil24_test.o

#sata_sil24_test-objs := sata_sil24.o

#obj-$(CONFIG_SATA_DWC)             += sata_dwc_460ex.o

#obj-$(CONFIG_SATA_HIGHBANK) += sata_highbank.o libahci.o

#obj-$(CONFIG_AHCI_BRCM)                   += ahci_brcm.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_CEVA)            += ahci_ceva.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_DA850)         += ahci_da850.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_DM816)        += ahci_dm816.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_IMX)              += ahci_imx.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_MTK)             += ahci_mtk.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_MVEBU)       += ahci_mvebu.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_OCTEON)     += ahci_octeon.o

#obj-$(CONFIG_AHCI_SUNXI) += ahci_sunxi.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_ST)                 += ahci_st.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_TEGRA)         += ahci_tegra.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_XGENE)         += ahci_xgene.o libahci.o libahci_platform.o

#obj-$(CONFIG_AHCI_QORIQ)         += ahci_qoriq.o libahci.o libahci_platform.o

# SFF w/ custom DMA

#obj-$(CONFIG_PDC_ADMA)           += pdc_adma.o

#obj-$(CONFIG_PATA_ARASAN_CF)         += pata_arasan_cf.o

#obj-$(CONFIG_PATA_OCTEON_CF)        += pata_octeon_cf.o

#obj-$(CONFIG_SATA_QSTOR)         += sata_qstor.o

#obj-$(CONFIG_SATA_SX4)                += sata_sx4.o

# SFF SATA w/ BMDMA

#obj-$(CONFIG_ATA_PIIX)                 += ata_piix.o

#obj-$(CONFIG_SATA_MV)                += sata_mv.o

#obj-$(CONFIG_SATA_NV)                 += sata_nv.o

#obj-$(CONFIG_SATA_PROMISE)     += sata_promise.o

#obj-$(CONFIG_SATA_RCAR)            += sata_rcar.o

#obj-$(CONFIG_SATA_SIL)                 += sata_sil.o

#obj-$(CONFIG_SATA_SIS)                 += sata_sis.o

#obj-$(CONFIG_SATA_SVW)              += sata_svw.o

#obj-$(CONFIG_SATA_ULI)                += sata_uli.o

#obj-$(CONFIG_SATA_VIA)                += sata_via.o

#obj-$(CONFIG_SATA_VITESSE)       += sata_vsc.o

# SFF PATA w/ BMDMA

#obj-$(CONFIG_PATA_ALI)                += pata_ali.o

#obj-$(CONFIG_PATA_AMD)             += pata_amd.o

#obj-$(CONFIG_PATA_ARTOP)         += pata_artop.o

#obj-$(CONFIG_PATA_ATIIXP) += pata_atiixp.o

#obj-$(CONFIG_PATA_ATP867X)      += pata_atp867x.o

#obj-$(CONFIG_PATA_BK3710)        += pata_bk3710.o

#obj-$(CONFIG_PATA_CMD64X)      += pata_cmd64x.o

#obj-$(CONFIG_PATA_CS5520)        += pata_cs5520.o

#obj-$(CONFIG_PATA_CS5530)        += pata_cs5530.o

#obj-$(CONFIG_PATA_CS5535)        += pata_cs5535.o

#obj-$(CONFIG_PATA_CS5536)        += pata_cs5536.o

#obj-$(CONFIG_PATA_CYPRESS)      += pata_cypress.o

#obj-$(CONFIG_PATA_EFAR)             += pata_efar.o

#obj-$(CONFIG_PATA_EP93XX)        += pata_ep93xx.o

#obj-$(CONFIG_PATA_FTIDE010)    += pata_ftide010.o

#obj-$(CONFIG_PATA_HPT366)        += pata_hpt366.o

#obj-$(CONFIG_PATA_HPT37X)        += pata_hpt37x.o

#obj-$(CONFIG_PATA_HPT3X2N)     += pata_hpt3x2n.o

#obj-$(CONFIG_PATA_HPT3X3)        += pata_hpt3x3.o

#obj-$(CONFIG_PATA_ICSIDE)          += pata_icside.o

#obj-$(CONFIG_PATA_IMX)               += pata_imx.o

#obj-$(CONFIG_PATA_IT8213)         += pata_it8213.o

#obj-$(CONFIG_PATA_IT821X)         += pata_it821x.o

#obj-$(CONFIG_PATA_JMICRON)    += pata_jmicron.o

#obj-$(CONFIG_PATA_MACIO)         += pata_macio.o

#obj-$(CONFIG_PATA_MARVELL)    += pata_marvell.o

#obj-$(CONFIG_PATA_MPC52xx)     += pata_mpc52xx.o

#obj-$(CONFIG_PATA_NETCELL)      += pata_netcell.o

#obj-$(CONFIG_PATA_NINJA32)      += pata_ninja32.o

#obj-$(CONFIG_PATA_NS87415)      += pata_ns87415.o

#obj-$(CONFIG_PATA_OLDPIIX)       += pata_oldpiix.o

#obj-$(CONFIG_PATA_OPTIDMA)    += pata_optidma.o

#obj-$(CONFIG_PATA_PDC2027X)   += pata_pdc2027x.o

#obj-$(CONFIG_PATA_PDC_OLD)    += pata_pdc202xx_old.o

#obj-$(CONFIG_PATA_RADISYS)      += pata_radisys.o

#obj-$(CONFIG_PATA_RDC)              += pata_rdc.o

#obj-$(CONFIG_PATA_SC1200)        += pata_sc1200.o

#obj-$(CONFIG_PATA_SCH)               += pata_sch.o

#obj-$(CONFIG_PATA_SERVERWORKS)   += pata_serverworks.o

#obj-$(CONFIG_PATA_SIL680) += pata_sil680.o

#obj-$(CONFIG_PATA_SIS)                 += pata_sis.o

#obj-$(CONFIG_PATA_TOSHIBA)     += pata_piccolo.o

#obj-$(CONFIG_PATA_TRIFLEX)       += pata_triflex.o

#obj-$(CONFIG_PATA_VIA)                += pata_via.o

#obj-$(CONFIG_PATA_WINBOND)  += pata_sl82c105.o

# SFF PIO only

#obj-$(CONFIG_PATA_CMD640_PCI)      += pata_cmd640.o

#obj-$(CONFIG_PATA_FALCON)       += pata_falcon.o

#obj-$(CONFIG_PATA_GAYLE) += pata_gayle.o

#obj-$(CONFIG_PATA_BUDDHA)     += pata_buddha.o

#obj-$(CONFIG_PATA_ISAPNP)         += pata_isapnp.o

#obj-$(CONFIG_PATA_IXP4XX_CF)  += pata_ixp4xx_cf.o

#obj-$(CONFIG_PATA_MPIIX)  += pata_mpiix.o

#obj-$(CONFIG_PATA_NS87410)      += pata_ns87410.o

#obj-$(CONFIG_PATA_OPTI)             += pata_opti.o

#obj-$(CONFIG_PATA_PCMCIA)       += pata_pcmcia.o

#obj-$(CONFIG_PATA_PALMLD)       += pata_palmld.o

#obj-$(CONFIG_PATA_PLATFORM)  += pata_platform.o

#obj-$(CONFIG_PATA_PLATFORM)  += pata_platform_test.o

#pata_platform_test-objs := pata_platform.o

#obj-$(CONFIG_PATA_OF_PLATFORM)    += pata_of_platform.o

#obj-$(CONFIG_PATA_OF_PLATFORM)    += pata_of_platform_test.o

#pata_of_platform_test-objs := pata_of_platform.o

#obj-$(CONFIG_PATA_RB532) += pata_rb532_cf.o

#obj-$(CONFIG_PATA_RZ1000)        += pata_rz1000.o

#obj-$(CONFIG_PATA_SAMSUNG_CF)     += pata_samsung_cf.o

#obj-$(CONFIG_PATA_PXA)               += pata_pxa.o

# Should be last but two libata driver

#obj-$(CONFIG_PATA_ACPI)             += pata_acpi.o

# Should be last but one libata driver

#obj-$(CONFIG_ATA_GENERIC)       += ata_generic.o

# Should be last libata driver

#obj-$(CONFIG_PATA_LEGACY)       += pata_legacy.o

#libata-y  := libata-core.o libata-scsi.o libata-eh.o \

#       libata-transport.o libata-trace.o

#libata-$(CONFIG_ATA_SFF)    += libata-sff.o

#libata-$(CONFIG_SATA_PMP)         += libata-pmp.o

#libata-$(CONFIG_ATA_ACPI) += libata-acpi.o

#libata-$(CONFIG_SATA_ZPODD)    += libata-zpodd.o

all: prepare

         $(MAKE) -C $(KERNELDIR) M=$(PWD) modules

         cp *.ko $(KERNEL)/

prepare:

         rm -rf $(KERNEL)

         mkdir $(KERNEL)

        

clean:

         rm -f *.o *.ko .*.cmd *.mod.*  *.unsigned *.order *.symvers .*.cmd.* .*.ko.* .*.mod.o.* .*.o.* *.mod

         rm -f -r .tmp_versions

         rm -rf $(KERNEL)

ahci驱动是一个pcie驱动,要使该驱动和我们的pcie设备匹配,需要在设备信息表ahci_pci_tbl[]中增加对应的FPGA pcie设备信息,我这里PCIE厂家id是0x7018,设备id是0x8080,修改如下所示:

{ PCI_DEVICE(0x7018, 0x8080), board_ahci},

由于系统本身已经安装了ahci.ko驱动因此还需要修改驱动名称,否则安装相同的驱动会报驱动已安装错误。如驱动名称为“ahci_test”修改ahci.c文件中宏定义如下:

#define DRV_NAME         "ahci_test"

完成以上修改后编译、安装驱动后/dev目录下依然未生成sata盘对应的设备节点文件。

dmesg命令查看内核打印信息发现未打印端口相关信息。通过查看驱动代码发现读取端口映射寄存器为0。读取端口映射寄存器代码如下:

手动将该寄存器0x7100000c(0x71000000为sata控制器bar基地址)置1后重新安装驱动,/dev/目录下依然未生成sata设备文件。dmesg查看内核驱动打印信息显示端口状态为link down。

查看端口状态寄存器0x71000128(0x71000100为端口0寄存器基地址)的值为0x30,说明端口sata设备确实没有link up。 将端口速率设置寄存器0x71080020设置成1,将sata3.0 改成sata2.0(该寄存器的值和端口速率对应关系为:0:sata1.0 ,1:sata2.0,2:sata3.0)。

重新安装驱动在/dev/目录下成功生成了sda设备文件。此时再次读取端口状态寄存器0x71000128值为0x123。

为了使驱动安装后自动配置以上两个寄存器,将以上两个寄存器配置在驱动中通过代码实现如下:

为了后续调试时能够隔离软件问题,梳理出以下初始化sata控制器步骤(memtool是内存读写工具):

1、进根权限

sudo su 输入密码

2、使能pcie设备

setpci -d 0731:8000 4.B=6

3、复位sata控制器HBA_HOST_GHC

memtool mw 0x71000004 0x01

4、设置HBA_HOST_CAP寄存器

memtool mw 0x71000000 0x6F36FF05

5、读取HBA_HOST_CAP寄存器

memtool md 0x71000000

6、设置端口映射寄存器HBA_HOST_PI

memtool mw 0x7100000c 0x1

7、HBA_PORT_CMD寄存器写0

memtool mw 0x71000118 0x0

8、HBA_HOST_BISTCR寄存器写0x5700

memtool mw 0x710000a4 0x5700

9、读取HBA_HOST_BISTCR寄存器写

memtool md 0x710000a4

10、读取HBA_PORT_CMD寄存器,读取的值为0x4

memtool md 0x71000118

11、HBA_PORT_CMD第4位置1

memtool mw 0x71000118 0x14

12、HBA_PORT_CMD第1位置1

memtool mw 0x71000118 0x16

13、读HBA_PORT_SSTS寄存器,寄存器第0位或者第1位需要置1才link up

memtool md 0x71000128

如果无法link up 需要执行完步骤1-6后设置0x71080020寄存器,将sata3.0 改成sata1.0

memtool mw 0x71080020 0x1

然后执行步骤7-13

参考手册《DWC_ahsata_databook.pdf》

简单梳理了几个关键的数据读写盘相关接口,后续有时间再深入研究。

        

struct ata_port_operations ahci_ops = {

         .inherits            = &sata_pmp_port_ops,

         .qc_defer          = ahci_pmp_qc_defer,

         .qc_prep           = ahci_qc_prep,

         .qc_issue          = ahci_qc_issue,

         .qc_fill_rtf                  = ahci_qc_fill_rtf,

         .freeze                        = ahci_freeze,

         .thaw                          = ahci_thaw,

         .softreset                   = ahci_softreset,

         .hardreset                 = ahci_hardreset,

         .postreset                  = ahci_postreset,

         .pmp_softreset                 = ahci_softreset,

         .error_handler                   = ahci_error_handler,

         .post_internal_cmd          = ahci_post_internal_cmd,

         .dev_config                = ahci_dev_config,

         .scr_read          = ahci_scr_read,

         .scr_write                  = ahci_scr_write,

         .pmp_attach             = ahci_pmp_attach,

         .pmp_detach            = ahci_pmp_detach,

         .set_lpm           = ahci_set_lpm,

         .em_show                  = ahci_led_show,

         .em_store                  = ahci_led_store,

         .sw_activity_show  = ahci_activity_show,

         .sw_activity_store  = ahci_activity_store,

         .transmit_led_message  = ahci_transmit_led_message,

#ifdef CONFIG_PM

         .port_suspend          = ahci_port_suspend,

         .port_resume           = ahci_port_resume,

#endif

         .port_start                = ahci_port_start,

         .port_stop                 = ahci_port_stop,

};

EXPORT_SYMBOL_GPL(ahci_ops);

#命令填充

         .qc_prep           = ahci_qc_prep,

#命令提交

         .qc_issue          = ahci_qc_issue,

        

#初始化sata控制器

ahci_pci_init_controller

#注册相关设备文件,生成/dev/sda设备

int ahci_host_activate(struct ata_host *host, struct scsi_host_template *sht)

有关FPGA SATA IP控制器的SATA接口调试记录的更多相关文章

  1. Ruby Readline 在向上箭头上使控制台崩溃 - 2

    当我在Rails控制台中按向上或向左箭头时,出现此错误:irb(main):001:0>/Users/me/.rvm/gems/ruby-2.0.0-p247/gems/rb-readline-0.4.2/lib/rbreadline.rb:4269:in`blockin_rl_dispatch_subseq':invalidbytesequenceinUTF-8(ArgumentError)我使用rvm来管理我的ruby​​安装。我正在使用=>ruby-2.0.0-p247[x86_64]我使用bundle来管理我的gem,并且我有rb-readline(0.4.2)(人们推荐的最少

  2. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  3. ruby-on-rails - 带 Spring 锁的 Rails 4 控制台 - 2

    我正在使用Ruby2.1.1和Rails4.1.0.rc1。当执行railsc时,它被锁定了。使用Ctrl-C停止,我得到以下错误日志:~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`gets':Interruptfrom~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`verify_server_version'from~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.

  4. ruby-on-rails - openshift 上的 rails 控制台 - 2

    我将我的Rails应用程序部署到OpenShift,它运行良好,但我无法在生产服务器上运行“Rails控制台”。它给了我这个错误。我该如何解决这个问题?我尝试更新ruby​​gems,但它也给出了权限被拒绝的错误,我也无法做到。railsc错误:Warning:You'reusingRubygems1.8.24withSpring.UpgradetoatleastRubygems2.1.0andrun`gempristine--all`forbetterstartupperformance./opt/rh/ruby193/root/usr/share/rubygems/rubygems

  5. ruby-on-rails - Rails 5 Active Record 记录无效错误 - 2

    我有两个Rails模型,即Invoice和Invoice_details。一个Invoice_details属于Invoice,一个Invoice有多个Invoice_details。我无法使用accepts_nested_attributes_forinInvoice通过Invoice模型保存Invoice_details。我收到以下错误:(0.2ms)BEGIN(0.2ms)ROLLBACKCompleted422UnprocessableEntityin25ms(ActiveRecord:4.0ms)ActiveRecord::RecordInvalid(Validationfa

  6. ruby-on-rails - 无法让 rspec、spork 和调试器正常运行 - 2

    GivenIamadumbprogrammerandIamusingrspecandIamusingsporkandIwanttodebug...mmm...let'ssaaay,aspecforPhone.那么,我应该把“require'ruby-debug'”行放在哪里,以便在phone_spec.rb的特定点停止处理?(我所要求的只是一个大而粗的箭头,即使是一个有挑战性的程序员也能看到:-3)我已经尝试了很多位置,除非我没有正确测试它们,否则会发生一些奇怪的事情:在spec_helper.rb中的以下位置:require'rubygems'require'spork'

  7. ruby - JetBrains RubyMine 3.2.4 调试器不工作 - 2

    使用Ruby1.9.2运行IDE提示说需要gemruby​​-debug-base19x并提供安装它。但是,在尝试安装它时会显示消息Failedtoinstallgems.Followinggemswerenotinstalled:C:/ProgramFiles(x86)/JetBrains/RubyMine3.2.4/rb/gems/ruby-debug-base19x-0.11.30.pre2.gem:Errorinstallingruby-debug-base19x-0.11.30.pre2.gem:The'linecache19'nativegemrequiresinstall

  8. ruby-on-rails - 如何调试 cucumber 测试? - 2

    我有:When/^(?:|I)follow"([^"]*)"(?:within"([^"]*)")?$/do|link,selector|with_scope(selector)doclick_link(link)endend我打电话的地方:Background:GivenIamanexistingadminuserWhenIfollow"CLIENTS"我的HTML是这样的:CLIENTS我一直收到这个错误:.F-.F--U-----U(::)failedsteps(::)nolinkwithtitle,idortext'CLIENTS'found(Capybara::Element

  9. postman接口测试工具-基础使用教程 - 2

    1.postman介绍Postman一款非常流行的API调试工具。其实,开发人员用的更多。因为测试人员做接口测试会有更多选择,例如Jmeter、soapUI等。不过,对于开发过程中去调试接口,Postman确实足够的简单方便,而且功能强大。2.下载安装官网地址:https://www.postman.com/下载完成后双击安装吧,安装过程极其简单,无需任何操作3.使用教程这里以百度为例,工具使用简单,填写URL地址即可发送请求,在下方查看响应结果和响应状态码常用方法都有支持请求方法:getpostputdeleteGet、Post、Put与Delete的作用get:请求方法一般是用于数据查询,

  10. C51单片机——实现用独立按键控制LED亮灭(调用函数篇) - 2

    说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时

随机推荐