sata_imx.c 1.46 KB
/*
 * Copyright 2017 NXP
 *
 * SPDX-License-Identifier:	GPL-2.0+
 */
#include <common.h>
#include <linux/errno.h>
#include <asm/io.h>
#include <asm/mach-imx/sci/sci.h>
#include <ahci.h>
#include <scsi.h>
#include <imx8_hsio.h>

int sata_init(void)
{
	int ret;
	u32 val, i = 0;

	printf("start sata init\n");
	writel(0x22222222, GPR_LPCG_PHYX2APB_0_APB);
	writel(0x22222222, GPR_LPCG_PHYX1_APB);

	setbits_le32(0x5F130008, BIT(21));
	setbits_le32(0x5F130008, BIT(23));

	/* PHY_MODE to SATA100Mhz ref clk */
	setbits_le32(HW_PHYX1_CTRL0_ADDR, BIT(19));

	/*
	 * bit 0 rx ena, bit 1 tx ena, bit 11 fast_init,
	 * bit12 PHY_X1_EPCS_SEL 1.
	 */
	setbits_le32(HW_MISC_CTRL0_ADDR, HW_MISC_CTRL0_IOB_RXENA
		     | HW_MISC_CTRL0_PHY_X1_EPCS_SEL);

	clrbits_le32(HW_SATA_CTRL0_ADDR, HW_SATA_CTRL0_PHY_RESET);
	setbits_le32(HW_SATA_CTRL0_ADDR, HW_SATA_CTRL0_PHY_RESET);
	setbits_le32(HW_SATA_CTRL0_ADDR, HW_SATA_CTRL0_RESET);
	udelay(1);
	clrbits_le32(HW_SATA_CTRL0_ADDR, HW_SATA_CTRL0_RESET);
	setbits_le32(HW_SATA_CTRL0_ADDR, HW_SATA_CTRL0_RESET);

	setbits_le32(HW_PHYX1_CTRL0_ADDR, HW_PHYX1_CTRL0_APB_RSTN);

	for (i = 0; i < 100; i++) {
		val = readl(HW_PHYX1_STTS0_ADDR);
		val &= HW_PHYX1_STTS0_LANE0_TX_PLL_LOCK;
		if (val == HW_PHYX1_STTS0_LANE0_TX_PLL_LOCK)
			break;
		udelay(1);
	}

	if (val != HW_PHYX1_STTS0_LANE0_TX_PLL_LOCK) {
		printf("TX PLL is not locked.\n");
		return -ENODEV;
	}

	ret = ahci_init((void __iomem *)AHCI_BASE_ADDR);
	if (ret)
		return ret;
	scsi_scan(1);

	return 0;
}