1171 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1171 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * (C) Copyright 2005
 | |
|  * Oxford Semiconductor Ltd
 | |
|  *
 | |
|  * See file CREDITS for list of people who contributed to this
 | |
|  * project.
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License as
 | |
|  * published by the Free Software Foundation; either version 2 of
 | |
|  * the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software
 | |
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,`
 | |
|  * MA 02111-1307 USA
 | |
|  */
 | |
| #include <common.h>
 | |
| #include <asm/arch/clock.h>
 | |
| 
 | |
| /**
 | |
|  * SATA related definitions
 | |
|  */
 | |
| #define ATA_PORT_CTL        0
 | |
| #define ATA_PORT_FEATURE    1
 | |
| #define ATA_PORT_NSECT      2
 | |
| #define ATA_PORT_LBAL       3
 | |
| #define ATA_PORT_LBAM       4
 | |
| #define ATA_PORT_LBAH       5
 | |
| #define ATA_PORT_DEVICE     6
 | |
| #define ATA_PORT_COMMAND    7
 | |
| 
 | |
| /* The offsets to the SATA registers */
 | |
| #define SATA_ORB1_OFF           0
 | |
| #define SATA_ORB2_OFF           1
 | |
| #define SATA_ORB3_OFF           2
 | |
| #define SATA_ORB4_OFF           3
 | |
| #define SATA_ORB5_OFF           4
 | |
| 
 | |
| #define SATA_FIS_ACCESS         11
 | |
| #define SATA_INT_STATUS_OFF     12  /* Read only */
 | |
| #define SATA_INT_CLR_OFF        12  /* Write only */
 | |
| #define SATA_INT_ENABLE_OFF     13  /* Read only */
 | |
| #define SATA_INT_ENABLE_SET_OFF 13  /* Write only */
 | |
| #define SATA_INT_ENABLE_CLR_OFF 14  /* Write only */
 | |
| #define SATA_VERSION_OFF        15
 | |
| #define SATA_CONTROL_OFF        23
 | |
| #define SATA_COMMAND_OFF        24
 | |
| #define SATA_PORT_CONTROL_OFF   25
 | |
| #define SATA_DRIVE_CONTROL_OFF  26
 | |
| 
 | |
| /* The offsets to the link registers that are access in an asynchronous manner */
 | |
| #define SATA_LINK_DATA     28
 | |
| #define SATA_LINK_RD_ADDR  29
 | |
| #define SATA_LINK_WR_ADDR  30
 | |
| #define SATA_LINK_CONTROL  31
 | |
| 
 | |
| /* SATA interrupt status register fields */
 | |
| #define SATA_INT_STATUS_EOC_RAW_BIT     ( 0 + 16)
 | |
| #define SATA_INT_STATUS_ERROR_BIT       ( 2 + 16)
 | |
| #define SATA_INT_STATUS_EOADT_RAW_BIT   ( 1 + 16)
 | |
| 
 | |
| /* SATA core command register commands */
 | |
| #define SATA_CMD_WRITE_TO_ORB_REGS              2
 | |
| #define SATA_CMD_WRITE_TO_ORB_REGS_NO_COMMAND   4
 | |
| 
 | |
| #define SATA_CMD_BUSY_BIT 7
 | |
| 
 | |
| #define SATA_SCTL_CLR_ERR 0x00000316UL
 | |
| 
 | |
| #define SATA_LBAL_BIT    0
 | |
| #define SATA_LBAM_BIT    8
 | |
| #define SATA_LBAH_BIT    16
 | |
| #define SATA_HOB_LBAH_BIT 24
 | |
| #define SATA_DEVICE_BIT  24
 | |
| #define SATA_NSECT_BIT   0
 | |
| #define SATA_HOB_NSECT_BIT   8
 | |
| #define SATA_LBA32_BIT       0
 | |
| #define SATA_LBA40_BIT       8
 | |
| #define SATA_FEATURE_BIT 16
 | |
| #define SATA_COMMAND_BIT 24
 | |
| #define SATA_CTL_BIT     24
 | |
| 
 | |
| /* ATA status (7) register field definitions */
 | |
| #define ATA_STATUS_BSY_BIT     7
 | |
| #define ATA_STATUS_DRDY_BIT    6
 | |
| #define ATA_STATUS_DF_BIT      5
 | |
| #define ATA_STATUS_DRQ_BIT     3
 | |
| #define ATA_STATUS_ERR_BIT     0
 | |
| 
 | |
| /* ATA device (6) register field definitions */
 | |
| #define ATA_DEVICE_FIXED_MASK 0xA0
 | |
| #define ATA_DEVICE_DRV_BIT 4
 | |
| #define ATA_DEVICE_DRV_NUM_BITS 1
 | |
| #define ATA_DEVICE_LBA_BIT 6
 | |
| 
 | |
| /* ATA Command register initiated commands */
 | |
| #define ATA_CMD_INIT    0x91
 | |
| #define ATA_CMD_IDENT   0xEC
 | |
| 
 | |
| #define SATA_STD_ASYNC_REGS_OFF 0x20
 | |
| #define SATA_SCR_STATUS      0
 | |
| #define SATA_SCR_ERROR       1
 | |
| #define SATA_SCR_CONTROL     2
 | |
| #define SATA_SCR_ACTIVE      3
 | |
| #define SATA_SCR_NOTIFICAION 4
 | |
| 
 | |
| #define SATA_BURST_BUF_FORCE_EOT_BIT        0
 | |
| #define SATA_BURST_BUF_DATA_INJ_ENABLE_BIT  1
 | |
| #define SATA_BURST_BUF_DIR_BIT              2
 | |
| #define SATA_BURST_BUF_DATA_INJ_END_BIT     3
 | |
| #define SATA_BURST_BUF_FIFO_DIS_BIT         4
 | |
| #define SATA_BURST_BUF_DIS_DREQ_BIT         5
 | |
| #define SATA_BURST_BUF_DREQ_BIT             6
 | |
| 
 | |
| #define SATA_OPCODE_MASK 0x3
 | |
| 
 | |
| #define SATA_DMA_CHANNEL 0
 | |
| 
 | |
| #define DMA_CTRL_STATUS      (0x0)
 | |
| #define DMA_BASE_SRC_ADR     (0x4)
 | |
| #define DMA_BASE_DST_ADR     (0x8)
 | |
| #define DMA_BYTE_CNT         (0xC)
 | |
| #define DMA_CURRENT_SRC_ADR  (0x10)
 | |
| #define DMA_CURRENT_DST_ADR  (0x14)
 | |
| #define DMA_CURRENT_BYTE_CNT (0x18)
 | |
| #define DMA_INTR_ID          (0x1C)
 | |
| #define DMA_INTR_CLEAR_REG   (DMA_CURRENT_SRC_ADR)
 | |
| 
 | |
| #define DMA_CALC_REG_ADR(channel, register) ((volatile u32*)(DMA_BASE + ((channel) << 5) + (register)))
 | |
| 
 | |
| #define DMA_CTRL_STATUS_FAIR_SHARE_ARB            (1 << 0)
 | |
| #define DMA_CTRL_STATUS_IN_PROGRESS               (1 << 1)
 | |
| #define DMA_CTRL_STATUS_SRC_DREQ_MASK             (0x0000003C)
 | |
| #define DMA_CTRL_STATUS_SRC_DREQ_SHIFT            (2)
 | |
| #define DMA_CTRL_STATUS_DEST_DREQ_MASK            (0x000003C0)
 | |
| #define DMA_CTRL_STATUS_DEST_DREQ_SHIFT           (6)
 | |
| #define DMA_CTRL_STATUS_INTR                      (1 << 10)
 | |
| #define DMA_CTRL_STATUS_NXT_FREE                  (1 << 11)
 | |
| #define DMA_CTRL_STATUS_RESET                     (1 << 12)
 | |
| #define DMA_CTRL_STATUS_DIR_MASK                  (0x00006000)
 | |
| #define DMA_CTRL_STATUS_DIR_SHIFT                 (13)
 | |
| #define DMA_CTRL_STATUS_SRC_ADR_MODE              (1 << 15)
 | |
| #define DMA_CTRL_STATUS_DEST_ADR_MODE             (1 << 16)
 | |
| #define DMA_CTRL_STATUS_TRANSFER_MODE_A           (1 << 17)
 | |
| #define DMA_CTRL_STATUS_TRANSFER_MODE_B           (1 << 18)
 | |
| #define DMA_CTRL_STATUS_SRC_WIDTH_MASK            (0x00380000)
 | |
| #define DMA_CTRL_STATUS_SRC_WIDTH_SHIFT           (19)
 | |
| #define DMA_CTRL_STATUS_DEST_WIDTH_MASK           (0x01C00000)
 | |
| #define DMA_CTRL_STATUS_DEST_WIDTH_SHIFT          (22)
 | |
| #define DMA_CTRL_STATUS_PAUSE                     (1 << 25)
 | |
| #define DMA_CTRL_STATUS_INTERRUPT_ENABLE          (1 << 26)
 | |
| #define DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED      (1 << 27)
 | |
| #define DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED (1 << 28)
 | |
| #define DMA_CTRL_STATUS_STARVE_LOW_PRIORITY       (1 << 29)
 | |
| #define DMA_CTRL_STATUS_INTR_CLEAR_ENABLE         (1 << 30)
 | |
| 
 | |
| #define DMA_BYTE_CNT_MASK        ((1 << 21) - 1)
 | |
| #define DMA_BYTE_CNT_WR_EOT_MASK (1 << 30)
 | |
| #define DMA_BYTE_CNT_RD_EOT_MASK (1 << 31)
 | |
| #define DMA_BYTE_CNT_BURST_MASK  (1 << 28)
 | |
| 
 | |
| #define MAKE_FIELD(value, num_bits, bit_num) (((value) & ((1 << (num_bits)) - 1)) << (bit_num))
 | |
| 
 | |
| typedef enum oxnas_dma_mode {
 | |
| 	OXNAS_DMA_MODE_FIXED, OXNAS_DMA_MODE_INC
 | |
| } oxnas_dma_mode_t;
 | |
| 
 | |
| typedef enum oxnas_dma_direction {
 | |
| 	OXNAS_DMA_TO_DEVICE, OXNAS_DMA_FROM_DEVICE
 | |
| } oxnas_dma_direction_t;
 | |
| 
 | |
| /* The available buses to which the DMA controller is attached */
 | |
| typedef enum oxnas_dma_transfer_bus {
 | |
| 	OXNAS_DMA_SIDE_A, OXNAS_DMA_SIDE_B
 | |
| } oxnas_dma_transfer_bus_t;
 | |
| 
 | |
| /* Direction of data flow between the DMA controller's pair of interfaces */
 | |
| typedef enum oxnas_dma_transfer_direction {
 | |
| 	OXNAS_DMA_A_TO_A, OXNAS_DMA_B_TO_A, OXNAS_DMA_A_TO_B, OXNAS_DMA_B_TO_B
 | |
| } oxnas_dma_transfer_direction_t;
 | |
| 
 | |
| /* The available data widths */
 | |
| typedef enum oxnas_dma_transfer_width {
 | |
| 	OXNAS_DMA_TRANSFER_WIDTH_8BITS,
 | |
| 	OXNAS_DMA_TRANSFER_WIDTH_16BITS,
 | |
| 	OXNAS_DMA_TRANSFER_WIDTH_32BITS
 | |
| } oxnas_dma_transfer_width_t;
 | |
| 
 | |
| /* The mode of the DMA transfer */
 | |
| typedef enum oxnas_dma_transfer_mode {
 | |
| 	OXNAS_DMA_TRANSFER_MODE_SINGLE, OXNAS_DMA_TRANSFER_MODE_BURST
 | |
| } oxnas_dma_transfer_mode_t;
 | |
| 
 | |
| /* The available transfer targets */
 | |
| typedef enum oxnas_dma_dreq {
 | |
| 	OXNAS_DMA_DREQ_SATA = 0, OXNAS_DMA_DREQ_MEMORY = 15
 | |
| } oxnas_dma_dreq_t;
 | |
| 
 | |
| typedef struct oxnas_dma_device_settings {
 | |
| 	unsigned long address_;
 | |
| 	unsigned fifo_size_; // Chained transfers must take account of FIFO offset at end of previous transfer
 | |
| 	unsigned char dreq_;
 | |
| 	unsigned read_eot_ :1;
 | |
| 	unsigned read_final_eot_ :1;
 | |
| 	unsigned write_eot_ :1;
 | |
| 	unsigned write_final_eot_ :1;
 | |
| 	unsigned bus_ :1;
 | |
| 	unsigned width_ :2;
 | |
| 	unsigned transfer_mode_ :1;
 | |
| 	unsigned address_mode_ :1;
 | |
| 	unsigned address_really_fixed_ :1;
 | |
| } oxnas_dma_device_settings_t;
 | |
| 
 | |
| static const int MAX_NO_ERROR_LOOPS = 100000; /* 1 second in units of 10uS */
 | |
| static const int MAX_DMA_XFER_LOOPS = 300000; /* 30 seconds in units of 100uS */
 | |
| static const int MAX_DMA_ABORT_LOOPS = 10000; /* 0.1 second in units of 10uS */
 | |
| static const int MAX_SRC_READ_LOOPS = 10000; /* 0.1 second in units of 10uS */
 | |
| static const int MAX_SRC_WRITE_LOOPS = 10000; /* 0.1 second in units of 10uS */
 | |
| static const int MAX_NOT_BUSY_LOOPS = 10000; /* 1 second in units of 100uS */
 | |
| 
 | |
| /* The internal SATA drive on which we should attempt to find partitions */
 | |
| static volatile u32* sata_regs_base[2] = { (volatile u32*) SATA_0_REGS_BASE,
 | |
| 		(volatile u32*) SATA_1_REGS_BASE,
 | |
| 
 | |
| };
 | |
| static u32 wr_sata_orb1[2] = { 0, 0 };
 | |
| static u32 wr_sata_orb2[2] = { 0, 0 };
 | |
| static u32 wr_sata_orb3[2] = { 0, 0 };
 | |
| static u32 wr_sata_orb4[2] = { 0, 0 };
 | |
| 
 | |
| #ifdef CONFIG_LBA48
 | |
| /* need keeping a record of NSECT LBAL LBAM LBAH ide_outb values for lba48 support */
 | |
| #define OUT_HISTORY_BASE	ATA_PORT_NSECT
 | |
| #define OUT_HISTORY_MAX		ATA_PORT_LBAH
 | |
| static unsigned char out_history[2][OUT_HISTORY_MAX - OUT_HISTORY_BASE + 1] = {};
 | |
| #endif
 | |
| 
 | |
| static oxnas_dma_device_settings_t oxnas_sata_dma_settings = { .address_ =
 | |
| 	SATA_DATA_BASE, .fifo_size_ = 16, .dreq_ = OXNAS_DMA_DREQ_SATA,
 | |
| 		.read_eot_ = 0, .read_final_eot_ = 1, .write_eot_ = 0,
 | |
| 		.write_final_eot_ = 1, .bus_ = OXNAS_DMA_SIDE_B, .width_ =
 | |
| 			OXNAS_DMA_TRANSFER_WIDTH_32BITS, .transfer_mode_ =
 | |
| 			OXNAS_DMA_TRANSFER_MODE_BURST, .address_mode_ =
 | |
| 			OXNAS_DMA_MODE_FIXED, .address_really_fixed_ = 0 };
 | |
| 
 | |
| oxnas_dma_device_settings_t oxnas_ram_dma_settings = { .address_ = 0,
 | |
| 		.fifo_size_ = 0, .dreq_ = OXNAS_DMA_DREQ_MEMORY, .read_eot_ = 1,
 | |
| 		.read_final_eot_ = 1, .write_eot_ = 1, .write_final_eot_ = 1,
 | |
| 		.bus_ = OXNAS_DMA_SIDE_A, .width_ =
 | |
| 			OXNAS_DMA_TRANSFER_WIDTH_32BITS, .transfer_mode_ =
 | |
| 			OXNAS_DMA_TRANSFER_MODE_BURST, .address_mode_ =
 | |
| 			OXNAS_DMA_MODE_FIXED, .address_really_fixed_ = 1 };
 | |
| 
 | |
| static void xfer_wr_shadow_to_orbs(int device)
 | |
| {
 | |
| 	*(sata_regs_base[device] + SATA_ORB1_OFF) = wr_sata_orb1[device];
 | |
| 	*(sata_regs_base[device] + SATA_ORB2_OFF) = wr_sata_orb2[device];
 | |
| 	*(sata_regs_base[device] + SATA_ORB3_OFF) = wr_sata_orb3[device];
 | |
| 	*(sata_regs_base[device] + SATA_ORB4_OFF) = wr_sata_orb4[device];
 | |
| }
 | |
| 
 | |
| static inline void device_select(int device)
 | |
| {
 | |
| 	/* master/slave has no meaning to SATA core */
 | |
| }
 | |
| 
 | |
| static int disk_present[CONFIG_SYS_IDE_MAXDEVICE];
 | |
| 
 | |
| #include <ata.h>
 | |
| 
 | |
| unsigned char ide_inb(int device, int port)
 | |
| {
 | |
| 	unsigned char val = 0;
 | |
| 
 | |
| 	/* Only permit accesses to disks found to be present during ide_preinit() */
 | |
| 	if (!disk_present[device]) {
 | |
| 		return ATA_STAT_FAULT;
 | |
| 	}
 | |
| 
 | |
| 	device_select(device);
 | |
| 
 | |
| 	switch (port) {
 | |
| 	case ATA_PORT_CTL:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB4_OFF)
 | |
| 			& (0xFFUL << SATA_CTL_BIT)) >> SATA_CTL_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_FEATURE:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
 | |
| 			& (0xFFUL << SATA_FEATURE_BIT)) >> SATA_FEATURE_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_NSECT:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
 | |
| 			& (0xFFUL << SATA_NSECT_BIT)) >> SATA_NSECT_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAL:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
 | |
| 			& (0xFFUL << SATA_LBAL_BIT)) >> SATA_LBAL_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAM:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
 | |
| 			& (0xFFUL << SATA_LBAM_BIT)) >> SATA_LBAM_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAH:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
 | |
| 			& (0xFFUL << SATA_LBAH_BIT)) >> SATA_LBAH_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_DEVICE:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
 | |
| 			& (0xFFUL << SATA_HOB_LBAH_BIT)) >> SATA_HOB_LBAH_BIT;
 | |
| 		val |= (*(sata_regs_base[device] + SATA_ORB1_OFF)
 | |
| 			& (0xFFUL << SATA_DEVICE_BIT)) >> SATA_DEVICE_BIT;
 | |
| 		break;
 | |
| 	case ATA_PORT_COMMAND:
 | |
| 		val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
 | |
| 			& (0xFFUL << SATA_COMMAND_BIT)) >> SATA_COMMAND_BIT;
 | |
| 		val |= ATA_STAT_DRQ;
 | |
| 		break;
 | |
| 	default:
 | |
| 		printf("ide_inb() Unknown port = %d\n", port);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	//    printf("inb: %d:%01x => %02x\n", device, port, val);
 | |
| 
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Possible that ATA status will not become no-error, so must have timeout
 | |
|  * @returns An int which is zero on error
 | |
|  */
 | |
| static inline int wait_no_error(int device)
 | |
| {
 | |
| 	int status = 0;
 | |
| 
 | |
| 	/* Check for ATA core error */
 | |
| 	if (*(sata_regs_base[device] + SATA_INT_STATUS_OFF)
 | |
| 		& (1 << SATA_INT_STATUS_ERROR_BIT)) {
 | |
| 		printf("wait_no_error() SATA core flagged error\n");
 | |
| 	} else {
 | |
| 		int loops = MAX_NO_ERROR_LOOPS;
 | |
| 		do {
 | |
| 			/* Check for ATA device error */
 | |
| 			if (!(ide_inb(device, ATA_PORT_COMMAND)
 | |
| 				& (1 << ATA_STATUS_ERR_BIT))) {
 | |
| 				status = 1;
 | |
| 				break;
 | |
| 			}
 | |
| 			udelay(10);
 | |
| 		} while (--loops);
 | |
| 
 | |
| 		if (!loops) {
 | |
| 			printf("wait_no_error() Timed out of wait for SATA no-error condition\n");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return status;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Expect SATA command to always finish, perhaps with error
 | |
|  * @returns An int which is zero on error
 | |
|  */
 | |
| static inline int wait_sata_command_not_busy(int device)
 | |
| {
 | |
| 	/* Wait for data to be available */
 | |
| 	int status = 0;
 | |
| 	int loops = MAX_NOT_BUSY_LOOPS;
 | |
| 	do {
 | |
| 		if (!(*(sata_regs_base[device] + SATA_COMMAND_OFF)
 | |
| 			& (1 << SATA_CMD_BUSY_BIT))) {
 | |
| 			status = 1;
 | |
| 			break;
 | |
| 		}
 | |
| 		udelay(100);
 | |
| 	} while (--loops);
 | |
| 
 | |
| 	if (!loops) {
 | |
| 		printf("wait_sata_command_not_busy() Timed out of wait for SATA command to finish\n");
 | |
| 	}
 | |
| 
 | |
| 	return status;
 | |
| }
 | |
| 
 | |
| void ide_outb(int device, int port, unsigned char val)
 | |
| {
 | |
| 	typedef enum send_method {
 | |
| 		SEND_NONE, SEND_SIMPLE, SEND_CMD, SEND_CTL,
 | |
| 	} send_method_t;
 | |
| 
 | |
| 	/* Only permit accesses to disks found to be present during ide_preinit() */
 | |
| 	if (!disk_present[device]) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	//    printf("outb: %d:%01x <= %02x\n", device, port, val);
 | |
| 
 | |
| 	device_select(device);
 | |
| 
 | |
| #ifdef CONFIG_LBA48
 | |
| 	if (port >= OUT_HISTORY_BASE && port <= OUT_HISTORY_MAX) {
 | |
| 		out_history[0][port - OUT_HISTORY_BASE] =
 | |
| 			out_history[1][port - OUT_HISTORY_BASE];
 | |
| 		out_history[1][port - OUT_HISTORY_BASE] = val;
 | |
| 	}
 | |
| #endif
 | |
| 	send_method_t send_regs = SEND_NONE;
 | |
| 	switch (port) {
 | |
| 	case ATA_PORT_CTL:
 | |
| 		wr_sata_orb4[device] &= ~(0xFFUL << SATA_CTL_BIT);
 | |
| 		wr_sata_orb4[device] |= (val << SATA_CTL_BIT);
 | |
| 		send_regs = SEND_CTL;
 | |
| 		break;
 | |
| 	case ATA_PORT_FEATURE:
 | |
| 		wr_sata_orb2[device] &= ~(0xFFUL << SATA_FEATURE_BIT);
 | |
| 		wr_sata_orb2[device] |= (val << SATA_FEATURE_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_NSECT:
 | |
| 		wr_sata_orb2[device] &= ~(0xFFUL << SATA_NSECT_BIT);
 | |
| 		wr_sata_orb2[device] |= (val << SATA_NSECT_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAL:
 | |
| 		wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAL_BIT);
 | |
| 		wr_sata_orb3[device] |= (val << SATA_LBAL_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAM:
 | |
| 		wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAM_BIT);
 | |
| 		wr_sata_orb3[device] |= (val << SATA_LBAM_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_LBAH:
 | |
| 		wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAH_BIT);
 | |
| 		wr_sata_orb3[device] |= (val << SATA_LBAH_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_DEVICE:
 | |
| 		wr_sata_orb1[device] &= ~(0xFFUL << SATA_DEVICE_BIT);
 | |
| 		wr_sata_orb1[device] |= (val << SATA_DEVICE_BIT);
 | |
| 		send_regs = SEND_SIMPLE;
 | |
| 		break;
 | |
| 	case ATA_PORT_COMMAND:
 | |
| 		wr_sata_orb2[device] &= ~(0xFFUL << SATA_COMMAND_BIT);
 | |
| 		wr_sata_orb2[device] |= (val << SATA_COMMAND_BIT);
 | |
| 		send_regs = SEND_CMD;
 | |
| #ifdef CONFIG_LBA48
 | |
| 		if (val == ATA_CMD_READ_EXT || val == ATA_CMD_WRITE_EXT)
 | |
| 		{
 | |
| 			/* fill high bytes of LBA48 && NSECT */
 | |
| 			wr_sata_orb2[device] &= ~(0xFFUL << SATA_HOB_NSECT_BIT);
 | |
| 			wr_sata_orb2[device] |=
 | |
| 				(out_history[0][ATA_PORT_NSECT - OUT_HISTORY_BASE] << SATA_HOB_NSECT_BIT);
 | |
| 
 | |
| 			wr_sata_orb3[device] &= ~(0xFFUL << SATA_HOB_LBAH_BIT);
 | |
| 			wr_sata_orb3[device] |=
 | |
| 				(out_history[0][ATA_PORT_LBAL - OUT_HISTORY_BASE] << SATA_HOB_LBAH_BIT);
 | |
| 
 | |
| 			wr_sata_orb4[device] &= ~(0xFFUL << SATA_LBA32_BIT);
 | |
| 			wr_sata_orb4[device] |=
 | |
| 				(out_history[0][ATA_PORT_LBAM - OUT_HISTORY_BASE] << SATA_LBA32_BIT);
 | |
| 
 | |
| 			wr_sata_orb4[device] &= ~(0xFFUL << SATA_LBA40_BIT);
 | |
| 			wr_sata_orb4[device] |=
 | |
| 				(out_history[0][ATA_PORT_LBAH - OUT_HISTORY_BASE] << SATA_LBA40_BIT);
 | |
| 		}
 | |
| #endif
 | |
| 		break;
 | |
| 	default:
 | |
| 		printf("ide_outb() Unknown port = %d\n", port);
 | |
| 	}
 | |
| 
 | |
| 	u32 command;
 | |
| 	switch (send_regs) {
 | |
| 	case SEND_CMD:
 | |
| 		wait_sata_command_not_busy(device);
 | |
| 		command = *(sata_regs_base[device] + SATA_COMMAND_OFF);
 | |
| 		command &= ~SATA_OPCODE_MASK;
 | |
| 		command |= SATA_CMD_WRITE_TO_ORB_REGS;
 | |
| 		xfer_wr_shadow_to_orbs(device);
 | |
| 		wait_sata_command_not_busy(device);
 | |
| 		*(sata_regs_base[device] + SATA_COMMAND_OFF) = command;
 | |
| 		if (!wait_no_error(device)) {
 | |
| 			printf("ide_outb() Wait for ATA no-error timed-out\n");
 | |
| 		}
 | |
| 		break;
 | |
| 	case SEND_CTL:
 | |
| 		wait_sata_command_not_busy(device);
 | |
| 		command = *(sata_regs_base[device] + SATA_COMMAND_OFF);
 | |
| 		command &= ~SATA_OPCODE_MASK;
 | |
| 		command |= SATA_CMD_WRITE_TO_ORB_REGS_NO_COMMAND;
 | |
| 		xfer_wr_shadow_to_orbs(device);
 | |
| 		wait_sata_command_not_busy(device);
 | |
| 		*(sata_regs_base[device] + SATA_COMMAND_OFF) = command;
 | |
| 		if (!wait_no_error(device)) {
 | |
| 			printf("ide_outb() Wait for ATA no-error timed-out\n");
 | |
| 		}
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static u32 encode_start(u32 ctrl_status)
 | |
| {
 | |
| 	return ctrl_status & ~DMA_CTRL_STATUS_PAUSE;
 | |
| }
 | |
| 
 | |
| /* start a paused DMA transfer in channel 0 of the SATA DMA core */
 | |
| static void dma_start(void)
 | |
| {
 | |
| 	unsigned int reg;
 | |
| 	reg = readl(SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
 | |
| 	reg = encode_start(reg);
 | |
| 	writel(reg, SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
 | |
| }
 | |
| 
 | |
| static unsigned long encode_control_status(
 | |
| 	oxnas_dma_device_settings_t* src_settings,
 | |
| 	oxnas_dma_device_settings_t* dst_settings)
 | |
| {
 | |
| 	unsigned long ctrl_status;
 | |
| 	oxnas_dma_transfer_direction_t direction;
 | |
| 
 | |
| 	ctrl_status = DMA_CTRL_STATUS_PAUSE;                           // Paused
 | |
| 	ctrl_status |= DMA_CTRL_STATUS_FAIR_SHARE_ARB;          // High priority
 | |
| 	ctrl_status |= (src_settings->dreq_ << DMA_CTRL_STATUS_SRC_DREQ_SHIFT); // Dreq
 | |
| 	ctrl_status |= (dst_settings->dreq_ << DMA_CTRL_STATUS_DEST_DREQ_SHIFT); // Dreq
 | |
| 	ctrl_status &= ~DMA_CTRL_STATUS_RESET;                         // !RESET
 | |
| 
 | |
| 	// Use new interrupt clearing register
 | |
| 	ctrl_status |= DMA_CTRL_STATUS_INTR_CLEAR_ENABLE;
 | |
| 
 | |
| 	// Setup the transfer direction and burst/single mode for the two DMA busses
 | |
| 	if (src_settings->bus_ == OXNAS_DMA_SIDE_A) {
 | |
| 		// Set the burst/single mode for bus A based on src device's settings
 | |
| 		if (src_settings->transfer_mode_
 | |
| 			== OXNAS_DMA_TRANSFER_MODE_BURST) {
 | |
| 			ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_A;
 | |
| 		} else {
 | |
| 			ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_A;
 | |
| 		}
 | |
| 
 | |
| 		if (dst_settings->bus_ == OXNAS_DMA_SIDE_A) {
 | |
| 			direction = OXNAS_DMA_A_TO_A;
 | |
| 		} else {
 | |
| 			direction = OXNAS_DMA_A_TO_B;
 | |
| 
 | |
| 			// Set the burst/single mode for bus B based on dst device's settings
 | |
| 			if (dst_settings->transfer_mode_
 | |
| 				== OXNAS_DMA_TRANSFER_MODE_BURST) {
 | |
| 				ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_B;
 | |
| 			} else {
 | |
| 				ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_B;
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Set the burst/single mode for bus B based on src device's settings
 | |
| 		if (src_settings->transfer_mode_
 | |
| 			== OXNAS_DMA_TRANSFER_MODE_BURST) {
 | |
| 			ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_B;
 | |
| 		} else {
 | |
| 			ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_B;
 | |
| 		}
 | |
| 
 | |
| 		if (dst_settings->bus_ == OXNAS_DMA_SIDE_A) {
 | |
| 			direction = OXNAS_DMA_B_TO_A;
 | |
| 
 | |
| 			// Set the burst/single mode for bus A based on dst device's settings
 | |
| 			if (dst_settings->transfer_mode_
 | |
| 				== OXNAS_DMA_TRANSFER_MODE_BURST) {
 | |
| 				ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_A;
 | |
| 			} else {
 | |
| 				ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_A;
 | |
| 			}
 | |
| 		} else {
 | |
| 			direction = OXNAS_DMA_B_TO_B;
 | |
| 		}
 | |
| 	}
 | |
| 	ctrl_status |= (direction << DMA_CTRL_STATUS_DIR_SHIFT);
 | |
| 
 | |
| 	// Setup source address mode fixed or increment
 | |
| 	if (src_settings->address_mode_ == OXNAS_DMA_MODE_FIXED) {
 | |
| 		// Fixed address
 | |
| 		ctrl_status &= ~(DMA_CTRL_STATUS_SRC_ADR_MODE);
 | |
| 
 | |
| 		// Set up whether fixed address is _really_ fixed
 | |
| 		if (src_settings->address_really_fixed_) {
 | |
| 			ctrl_status |= DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
 | |
| 		} else {
 | |
| 			ctrl_status &= ~DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Incrementing address
 | |
| 		ctrl_status |= DMA_CTRL_STATUS_SRC_ADR_MODE;
 | |
| 		ctrl_status &= ~DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
 | |
| 	}
 | |
| 
 | |
| 	// Setup destination address mode fixed or increment
 | |
| 	if (dst_settings->address_mode_ == OXNAS_DMA_MODE_FIXED) {
 | |
| 		// Fixed address
 | |
| 		ctrl_status &= ~(DMA_CTRL_STATUS_DEST_ADR_MODE);
 | |
| 
 | |
| 		// Set up whether fixed address is _really_ fixed
 | |
| 		if (dst_settings->address_really_fixed_) {
 | |
| 			ctrl_status |=
 | |
| 				DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
 | |
| 		} else {
 | |
| 			ctrl_status &=
 | |
| 				~DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Incrementing address
 | |
| 		ctrl_status |= DMA_CTRL_STATUS_DEST_ADR_MODE;
 | |
| 		ctrl_status &= ~DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
 | |
| 	}
 | |
| 
 | |
| 	// Set up the width of the transfers on the DMA buses
 | |
| 	ctrl_status |=
 | |
| 		(src_settings->width_ << DMA_CTRL_STATUS_SRC_WIDTH_SHIFT);
 | |
| 	ctrl_status |=
 | |
| 		(dst_settings->width_ << DMA_CTRL_STATUS_DEST_WIDTH_SHIFT);
 | |
| 
 | |
| 	// Setup the priority arbitration scheme
 | |
| 	ctrl_status &= ~DMA_CTRL_STATUS_STARVE_LOW_PRIORITY; // !Starve low priority
 | |
| 
 | |
| 	return ctrl_status;
 | |
| }
 | |
| 
 | |
| static u32 encode_final_eot(oxnas_dma_device_settings_t* src_settings,
 | |
| 				oxnas_dma_device_settings_t* dst_settings,
 | |
| 				unsigned long length)
 | |
| {
 | |
| 	// Write the length, with EOT configuration for a final transfer
 | |
| 	unsigned long encoded = length;
 | |
| 	if (dst_settings->write_final_eot_) {
 | |
| 		encoded |= DMA_BYTE_CNT_WR_EOT_MASK;
 | |
| 	} else {
 | |
| 		encoded &= ~DMA_BYTE_CNT_WR_EOT_MASK;
 | |
| 	}
 | |
| 	if (src_settings->read_final_eot_) {
 | |
| 		encoded |= DMA_BYTE_CNT_RD_EOT_MASK;
 | |
| 	} else {
 | |
| 		encoded &= ~DMA_BYTE_CNT_RD_EOT_MASK;
 | |
| 	}
 | |
| 	/*    if((src_settings->transfer_mode_) ||
 | |
| 	 (src_settings->transfer_mode_)) {
 | |
| 	 encoded |= DMA_BYTE_CNT_BURST_MASK;
 | |
| 	 } else {
 | |
| 	 encoded &= ~DMA_BYTE_CNT_BURST_MASK;
 | |
| 	 }*/
 | |
| 	return encoded;
 | |
| }
 | |
| 
 | |
| static void dma_start_write(const ulong* buffer, int num_bytes)
 | |
| {
 | |
| 	// Assemble complete memory settings
 | |
| 	oxnas_dma_device_settings_t mem_settings = oxnas_ram_dma_settings;
 | |
| 	mem_settings.address_ = (unsigned long) buffer;
 | |
| 	mem_settings.address_mode_ = OXNAS_DMA_MODE_INC;
 | |
| 
 | |
| 	writel(encode_control_status(&mem_settings, &oxnas_sata_dma_settings),
 | |
| 		SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
 | |
| 	writel(mem_settings.address_, SATA_DMA_REGS_BASE + DMA_BASE_SRC_ADR);
 | |
| 	writel(oxnas_sata_dma_settings.address_,
 | |
| 		SATA_DMA_REGS_BASE + DMA_BASE_DST_ADR);
 | |
| 	writel(encode_final_eot(&mem_settings, &oxnas_sata_dma_settings,
 | |
| 				num_bytes),
 | |
| 		SATA_DMA_REGS_BASE + DMA_BYTE_CNT);
 | |
| 
 | |
| 	dma_start();
 | |
| }
 | |
| 
 | |
| static void dma_start_read(ulong* buffer, int num_bytes)
 | |
| {
 | |
| 	// Assemble complete memory settings
 | |
| 	oxnas_dma_device_settings_t mem_settings = oxnas_ram_dma_settings;
 | |
| 	mem_settings.address_ = (unsigned long) buffer;
 | |
| 	mem_settings.address_mode_ = OXNAS_DMA_MODE_INC;
 | |
| 
 | |
| 	writel(encode_control_status(&oxnas_sata_dma_settings, &mem_settings),
 | |
| 		SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
 | |
| 	writel(oxnas_sata_dma_settings.address_,
 | |
| 		SATA_DMA_REGS_BASE + DMA_BASE_SRC_ADR);
 | |
| 	writel(mem_settings.address_, SATA_DMA_REGS_BASE + DMA_BASE_DST_ADR);
 | |
| 	writel(encode_final_eot(&oxnas_sata_dma_settings, &mem_settings,
 | |
| 				num_bytes),
 | |
| 		SATA_DMA_REGS_BASE + DMA_BYTE_CNT);
 | |
| 
 | |
| 	dma_start();
 | |
| }
 | |
| 
 | |
| static inline int dma_busy(void)
 | |
| {
 | |
| 	return readl(SATA_DMA_REGS_BASE + DMA_CTRL_STATUS)
 | |
| 		& DMA_CTRL_STATUS_IN_PROGRESS;
 | |
| }
 | |
| 
 | |
| static int wait_dma_not_busy(int device)
 | |
| {
 | |
| 	unsigned int cleanup_required = 0;
 | |
| 
 | |
| 	/* Poll for DMA completion */
 | |
| 	int loops = MAX_DMA_XFER_LOOPS;
 | |
| 	do {
 | |
| 		if (!dma_busy()) {
 | |
| 			break;
 | |
| 		}
 | |
| 		udelay(100);
 | |
| 	} while (--loops);
 | |
| 
 | |
| 	if (!loops) {
 | |
| 		printf("wait_dma_not_busy() Timed out of wait for DMA not busy\n");
 | |
| 		cleanup_required = 1;
 | |
| 	}
 | |
| 
 | |
| 	if (cleanup_required) {
 | |
| 		/* Abort DMA to make sure it has finished. */
 | |
| 		unsigned int ctrl_status = readl(
 | |
| 			SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
 | |
| 		ctrl_status |= DMA_CTRL_STATUS_RESET;
 | |
| 		writel(ctrl_status, SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
 | |
| 
 | |
| 		// Wait for the channel to become idle - should be quick as should
 | |
| 		// finish after the next AHB single or burst transfer
 | |
| 		loops = MAX_DMA_ABORT_LOOPS;
 | |
| 		do {
 | |
| 			if (!dma_busy()) {
 | |
| 				break;
 | |
| 			}
 | |
| 			udelay(10);
 | |
| 		} while (--loops);
 | |
| 
 | |
| 		if (!loops) {
 | |
| 			printf("wait_dma_not_busy() Timed out of wait for DMA channel abort\n");
 | |
| 		} else {
 | |
| 			/* Successfully cleanup the DMA channel */
 | |
| 			cleanup_required = 0;
 | |
| 		}
 | |
| 
 | |
| 		// Deassert reset for the channel
 | |
| 		ctrl_status = readl(SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
 | |
| 		ctrl_status &= ~DMA_CTRL_STATUS_RESET;
 | |
| 		writel(ctrl_status, SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
 | |
| 	}
 | |
| 
 | |
| 	return !cleanup_required;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Possible that ATA status will not become not-busy, so must have timeout
 | |
|  */
 | |
| static unsigned int wait_not_busy(int device, unsigned long timeout_secs)
 | |
| {
 | |
| 	int busy = 1;
 | |
| 	unsigned long loops = (timeout_secs * 1000) / 50;
 | |
| 	do {
 | |
| 		// Test the ATA status register BUSY flag
 | |
| 		if (!((*(sata_regs_base[device] + SATA_ORB2_OFF)
 | |
| 			>> SATA_COMMAND_BIT) & (1UL << ATA_STATUS_BSY_BIT))) {
 | |
| 			/* Not busy, so stop polling */
 | |
| 			busy = 0;
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		// Wait for 50mS before sampling ATA status register again
 | |
| 		udelay(50000);
 | |
| 	} while (--loops);
 | |
| 
 | |
| 	return busy;
 | |
| }
 | |
| 
 | |
| void ide_output_data(int device, const ulong *sect_buf, int words)
 | |
| {
 | |
| 	/* Only permit accesses to disks found to be present during ide_preinit() */
 | |
| 	if (!disk_present[device]) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* Select the required internal SATA drive */
 | |
| 	device_select(device);
 | |
| 
 | |
| 	/* Start the DMA channel sending data from the passed buffer to the SATA core */
 | |
| 	dma_start_write(sect_buf, words << 2);
 | |
| 
 | |
| 	/* Don't know why we need this delay, but without it the wait for DMA not
 | |
| 	 busy times soemtimes out, e.g. when saving environment to second disk */
 | |
| 	udelay(1000);
 | |
| 
 | |
| 	/* Wait for DMA to finish */
 | |
| 	if (!wait_dma_not_busy(device)) {
 | |
| 		printf("Timed out of wait for DMA channel for SATA device %d to have in-progress clear\n",
 | |
| 			device);
 | |
| 	}
 | |
| 
 | |
| 	/* Sata core should finish after DMA */
 | |
| 	if (wait_not_busy(device, 30)) {
 | |
| 		printf("Timed out of wait for SATA device %d to have BUSY clear\n",
 | |
| 			device);
 | |
| 	}
 | |
| 	if (!wait_no_error(device)) {
 | |
| 		printf("oxnas_sata_output_data() Wait for ATA no-error timed-out\n");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| #define SATA_DM_DBG1			(SATA_HOST_REGS_BASE + 0)
 | |
| #define SATA_DATACOUNT_PORT0		(SATA_HOST_REGS_BASE + 0x10)
 | |
| #define SATA_DATACOUNT_PORT1		(SATA_HOST_REGS_BASE + 0x14)
 | |
| #define SATA_DATA_MUX_RAM0		(SATA_HOST_REGS_BASE + 0x8000)
 | |
| #define SATA_DATA_MUX_RAM1		(SATA_HOST_REGS_BASE + 0xA000)
 | |
| /* Sata core debug1 register bits */
 | |
| #define SATA_CORE_PORT0_DATA_DIR_BIT	20
 | |
| #define SATA_CORE_PORT1_DATA_DIR_BIT	21
 | |
| #define SATA_CORE_PORT0_DATA_DIR	(1 << SATA_CORE_PORT0_DATA_DIR_BIT)
 | |
| #define SATA_CORE_PORT1_DATA_DIR	(1 << SATA_CORE_PORT1_DATA_DIR_BIT)
 | |
| 
 | |
| /**
 | |
|  * Ref bug-6320
 | |
|  *
 | |
|  * This code is a work around for a DMA hardware bug that will repeat the
 | |
|  * penultimate 8-bytes on some reads. This code will check that the amount
 | |
|  * of data transferred is a multiple of 512 bytes, if not the in it will
 | |
|  * fetch the correct data from a buffer in the SATA core and copy it into
 | |
|  * memory.
 | |
|  *
 | |
|  */
 | |
| static void sata_bug_6320_workaround(int port, ulong *candidate)
 | |
| {
 | |
| 	int is_read;
 | |
| 	int quads_transferred;
 | |
| 	int remainder;
 | |
| 	int sector_quads_remaining;
 | |
| 
 | |
| 	/* Only want to apply fix to reads */
 | |
| 	is_read = !(*((unsigned long*) SATA_DM_DBG1)
 | |
| 		& (port ? SATA_CORE_PORT1_DATA_DIR : SATA_CORE_PORT0_DATA_DIR));
 | |
| 
 | |
| 	/* Check for an incomplete transfer, i.e. not a multiple of 512 bytes
 | |
| 	 transferred (datacount_port register counts quads transferred) */
 | |
| 	quads_transferred = *((unsigned long*) (
 | |
| 		port ? SATA_DATACOUNT_PORT1 : SATA_DATACOUNT_PORT0));
 | |
| 
 | |
| 	remainder = quads_transferred & 0x7f;
 | |
| 	sector_quads_remaining = remainder ? (0x80 - remainder) : 0;
 | |
| 
 | |
| 	if (is_read && (sector_quads_remaining == 2)) {
 | |
| 		debug("SATA read fixup, only transfered %d quads, "
 | |
| 			"sector_quads_remaining %d, port %d\n",
 | |
| 			quads_transferred, sector_quads_remaining, port);
 | |
| 
 | |
| 		int total_len = ATA_SECT_SIZE;
 | |
| 		ulong *sata_data_ptr = (void*) (
 | |
| 			port ? SATA_DATA_MUX_RAM1 : SATA_DATA_MUX_RAM0)
 | |
| 			+ ((total_len - 8) % 2048);
 | |
| 
 | |
| 		*candidate = *sata_data_ptr;
 | |
| 		*(candidate + 1) = *(sata_data_ptr + 1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| void ide_input_data(int device, ulong *sect_buf, int words)
 | |
| {
 | |
| 	/* Only permit accesses to disks found to be present during ide_preinit() */
 | |
| 	if (!disk_present[device]) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* Select the required internal SATA drive */
 | |
| 	device_select(device);
 | |
| 
 | |
| 	/* Start the DMA channel receiving data from the SATA core into the passed buffer */
 | |
| 	dma_start_read(sect_buf, words << 2);
 | |
| 
 | |
| 	/* Sata core should finish before DMA */
 | |
| 	if (wait_not_busy(device, 30)) {
 | |
| 		printf("Timed out of wait for SATA device %d to have BUSY clear\n",
 | |
| 			device);
 | |
| 	}
 | |
| 	if (!wait_no_error(device)) {
 | |
| 		printf("oxnas_sata_output_data() Wait for ATA no-error timed-out\n");
 | |
| 	}
 | |
| 
 | |
| 	/* Wait for DMA to finish */
 | |
| 	if (!wait_dma_not_busy(device)) {
 | |
| 		printf("Timed out of wait for DMA channel for SATA device %d to have in-progress clear\n",
 | |
| 			device);
 | |
| 	}
 | |
| 
 | |
| 	if (words == ATA_SECTORWORDS)
 | |
| 		sata_bug_6320_workaround(device, sect_buf + words - 2);
 | |
| }
 | |
| 
 | |
| static u32 scr_read(int device, unsigned int sc_reg)
 | |
| {
 | |
| 	/* Setup adr of required register. std regs start eight into async region */
 | |
| 	*(sata_regs_base[device] + SATA_LINK_RD_ADDR) = sc_reg
 | |
| 		* 4+ SATA_STD_ASYNC_REGS_OFF;
 | |
| 
 | |
| 	/* Wait for data to be available */
 | |
| 	int loops = MAX_SRC_READ_LOOPS;
 | |
| 	do {
 | |
| 		if (*(sata_regs_base[device] + SATA_LINK_CONTROL) & 1UL) {
 | |
| 			break;
 | |
| 		}
 | |
| 		udelay(10);
 | |
| 	} while (--loops);
 | |
| 
 | |
| 	if (!loops) {
 | |
| 		printf("scr_read() Timed out of wait for read completion\n");
 | |
| 	}
 | |
| 
 | |
| 	/* Read the data from the async register */
 | |
| 	return *(sata_regs_base[device] + SATA_LINK_DATA);
 | |
| }
 | |
| 
 | |
| static void scr_write(int device, unsigned int sc_reg, u32 val)
 | |
| {
 | |
| 	/* Setup the data for the write */
 | |
| 	*(sata_regs_base[device] + SATA_LINK_DATA) = val;
 | |
| 
 | |
| 	/* Setup adr of required register. std regs start eight into async region */
 | |
| 	*(sata_regs_base[device] + SATA_LINK_WR_ADDR) = sc_reg
 | |
| 		* 4+ SATA_STD_ASYNC_REGS_OFF;
 | |
| 
 | |
| 	/* Wait for data to be written */
 | |
| 	int loops = MAX_SRC_WRITE_LOOPS;
 | |
| 	do {
 | |
| 		if (*(sata_regs_base[device] + SATA_LINK_CONTROL) & 1UL) {
 | |
| 			break;
 | |
| 		}
 | |
| 		udelay(10);
 | |
| 	} while (--loops);
 | |
| 
 | |
| 	if (!loops) {
 | |
| 		printf("scr_write() Timed out of wait for write completion\n");
 | |
| 	}
 | |
| }
 | |
| extern void workaround5458(void);
 | |
| 
 | |
| #define PHY_LOOP_COUNT  25  /* Wait for upto 5 seconds for PHY to be found */
 | |
| #define LOS_AND_TX_LVL   0x2988
 | |
| #define TX_ATTEN         0x55629
 | |
| 
 | |
| static int phy_reset(int device)
 | |
| {
 | |
| 	int phy_status = 0;
 | |
| 	int loops = 0;
 | |
| 
 | |
| 	scr_write(device, (0x60 - SATA_STD_ASYNC_REGS_OFF) / 4, LOS_AND_TX_LVL);
 | |
| 	scr_write(device, (0x70 - SATA_STD_ASYNC_REGS_OFF) / 4, TX_ATTEN);
 | |
| 
 | |
| 	/* limit it to Gen-1 SATA (1.5G) */
 | |
| 	scr_write(device, SATA_SCR_CONTROL, 0x311); /* Issue phy wake & core reset */
 | |
| 	scr_read(device, SATA_SCR_STATUS); /* Dummy read; flush */
 | |
| 	udelay(1000);
 | |
| 	scr_write(device, SATA_SCR_CONTROL, 0x310); /* Issue phy wake & clear core reset */
 | |
| 
 | |
| 	/* Wait for upto 5 seconds for PHY to become ready */
 | |
| 	do {
 | |
| 		udelay(200000);
 | |
| 		if ((scr_read(device, SATA_SCR_STATUS) & 0xf) == 3) {
 | |
| 			scr_write(device, SATA_SCR_ERROR, ~0);
 | |
| 			phy_status = 1;
 | |
| 			break;
 | |
| 		}
 | |
| 		//printf("No SATA PHY found status:0x%x\n", scr_read(device, SATA_SCR_STATUS));
 | |
| 	} while (++loops < PHY_LOOP_COUNT);
 | |
| 
 | |
| 	if (phy_status) {
 | |
| 		udelay(500000); /* wait half a second */
 | |
| 	}
 | |
| 
 | |
| 	return phy_status;
 | |
| }
 | |
| 
 | |
| #define FIS_LOOP_COUNT  25  /* Wait for upto 5 seconds for FIS to be received */
 | |
| static int wait_FIS(int device)
 | |
| {
 | |
| 	int status = 0;
 | |
| 	int loops = 0;
 | |
| 
 | |
| 	do {
 | |
| 		udelay(200000);
 | |
| 		if (ide_inb(device, ATA_PORT_NSECT) > 0) {
 | |
| 			status = 1;
 | |
| 			break;
 | |
| 		}
 | |
| 	} while (++loops < FIS_LOOP_COUNT);
 | |
| 
 | |
| 	return status;
 | |
| }
 | |
| 
 | |
| 
 | |
| #define SATA_PHY_ASIC_STAT  (0x44900000)
 | |
| #define SATA_PHY_ASIC_DATA  (0x44900004)
 | |
| 
 | |
| /**
 | |
|  * initialise functions and macros for ASIC implementation
 | |
|  */
 | |
| #define PH_GAIN         2
 | |
| #define FR_GAIN         3
 | |
| #define PH_GAIN_OFFSET  6
 | |
| #define FR_GAIN_OFFSET  8
 | |
| #define PH_GAIN_MASK  (0x3 << PH_GAIN_OFFSET)
 | |
| #define FR_GAIN_MASK  (0x3 << FR_GAIN_OFFSET)
 | |
| #define USE_INT_SETTING  (1<<5)
 | |
| 
 | |
| #define CR_READ_ENABLE  (1<<16)
 | |
| #define CR_WRITE_ENABLE (1<<17)
 | |
| #define CR_CAP_DATA     (1<<18)
 | |
| 
 | |
| static void wait_cr_ack(void)
 | |
| {
 | |
| 	while ((readl(SATA_PHY_ASIC_STAT) >> 16) & 0x1f)
 | |
| 		/* wait for an ack bit to be set */;
 | |
| }
 | |
| 
 | |
| static unsigned short read_cr(unsigned short address)
 | |
| {
 | |
| 	writel(address, SATA_PHY_ASIC_STAT);
 | |
| 	wait_cr_ack();
 | |
| 	writel(CR_READ_ENABLE, SATA_PHY_ASIC_DATA);
 | |
| 	wait_cr_ack();
 | |
| 	return readl(SATA_PHY_ASIC_STAT);
 | |
| }
 | |
| 
 | |
| static void write_cr(unsigned short data, unsigned short address)
 | |
| {
 | |
| 	writel(address, SATA_PHY_ASIC_STAT);
 | |
| 	wait_cr_ack();
 | |
| 	writel((data | CR_CAP_DATA), SATA_PHY_ASIC_DATA);
 | |
| 	wait_cr_ack();
 | |
| 	writel(CR_WRITE_ENABLE, SATA_PHY_ASIC_DATA);
 | |
| 	wait_cr_ack();
 | |
| 	return;
 | |
| }
 | |
| 
 | |
| void workaround5458(void)
 | |
| {
 | |
| 	unsigned i;
 | |
| 
 | |
| 	for (i = 0; i < 2; i++) {
 | |
| 		unsigned short rx_control = read_cr(0x201d + (i << 8));
 | |
| 		rx_control &= ~(PH_GAIN_MASK | FR_GAIN_MASK);
 | |
| 		rx_control |= PH_GAIN << PH_GAIN_OFFSET;
 | |
| 		rx_control |= FR_GAIN << FR_GAIN_OFFSET;
 | |
| 		rx_control |= USE_INT_SETTING;
 | |
| 		write_cr(rx_control, 0x201d + (i << 8));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int ide_preinit(void)
 | |
| {
 | |
| 	int num_disks_found = 0;
 | |
| 
 | |
| 	/* Initialise records of which disks are present to all present */
 | |
| 	int i;
 | |
| 	for (i = 0; i < CONFIG_SYS_IDE_MAXDEVICE; i++) {
 | |
| 		disk_present[i] = 1;
 | |
| 	}
 | |
| 
 | |
| 	/* Block reset SATA and DMA cores */
 | |
| 	reset_block(SYS_CTRL_RST_SATA, 1);
 | |
| 	reset_block(SYS_CTRL_RST_SATA_LINK, 1);
 | |
| 	reset_block(SYS_CTRL_RST_SATA_PHY, 1);
 | |
| 	reset_block(SYS_CTRL_RST_SGDMA, 1);
 | |
| 
 | |
| 	/* Enable clocks to SATA and DMA cores */
 | |
| 	enable_clock(SYS_CTRL_CLK_SATA);
 | |
| 	enable_clock(SYS_CTRL_CLK_DMA);
 | |
| 
 | |
| 	udelay(5000);
 | |
| 	reset_block(SYS_CTRL_RST_SATA_PHY, 0);
 | |
| 	udelay(50);
 | |
| 	reset_block(SYS_CTRL_RST_SATA, 0);
 | |
| 	reset_block(SYS_CTRL_RST_SATA_LINK, 0);
 | |
| 	udelay(50);
 | |
| 	reset_block(SYS_CTRL_RST_SGDMA, 0);
 | |
| 	udelay(100);
 | |
| 	/* Apply the Synopsis SATA PHY workarounds */
 | |
| 	workaround5458();
 | |
| 	udelay(10000);
 | |
| 
 | |
| 	/* disable and clear core interrupts */
 | |
| 	*((unsigned long*) SATA_HOST_REGS_BASE + SATA_INT_ENABLE_CLR_OFF) =
 | |
| 		~0UL;
 | |
| 	*((unsigned long*) SATA_HOST_REGS_BASE + SATA_INT_CLR_OFF) = ~0UL;
 | |
| 
 | |
| 	int device;
 | |
| 	for (device = 0; device < CONFIG_SYS_IDE_MAXDEVICE; device++) {
 | |
| 		int found = 0;
 | |
| 		int retries = 1;
 | |
| 
 | |
| 		/* Disable SATA interrupts */
 | |
| 		*(sata_regs_base[device] + SATA_INT_ENABLE_CLR_OFF) = ~0UL;
 | |
| 
 | |
| 		/* Clear any pending SATA interrupts */
 | |
| 		*(sata_regs_base[device] + SATA_INT_CLR_OFF) = ~0UL;
 | |
| 
 | |
| 		do {
 | |
| 			/* clear sector count register for FIS detection */
 | |
| 			ide_outb(device, ATA_PORT_NSECT, 0);
 | |
| 
 | |
| 			/* Get the PHY working */
 | |
| 			if (!phy_reset(device)) {
 | |
| 				printf("SATA PHY not ready for device %d\n",
 | |
| 					device);
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			if (!wait_FIS(device)) {
 | |
| 				printf("No FIS received from device %d\n",
 | |
| 					device);
 | |
| 			} else {
 | |
| 				if ((scr_read(device, SATA_SCR_STATUS) & 0xf)
 | |
| 					== 0x3) {
 | |
| 					if (wait_not_busy(device, 30)) {
 | |
| 						printf("Timed out of wait for SATA device %d to have BUSY clear\n",
 | |
| 							device);
 | |
| 					} else {
 | |
| 						++num_disks_found;
 | |
| 						found = 1;
 | |
| 					}
 | |
| 				} else {
 | |
| 					printf("No SATA device %d found, PHY status = 0x%08x\n",
 | |
| 						device,
 | |
| 						scr_read(
 | |
| 							device,
 | |
| 							SATA_SCR_STATUS));
 | |
| 				}
 | |
| 				break;
 | |
| 			}
 | |
| 		} while (retries--);
 | |
| 
 | |
| 		/* Record whether disk is present, so won't attempt to access it later */
 | |
| 		disk_present[device] = found;
 | |
| 	}
 | |
| 
 | |
| 	/* post disk detection clean-up */
 | |
| 	for (device = 0; device < CONFIG_SYS_IDE_MAXDEVICE; device++) {
 | |
| 		if (disk_present[device]) {
 | |
| 			/* set as ata-5 (28-bit) */
 | |
| 			*(sata_regs_base[device] + SATA_DRIVE_CONTROL_OFF) =
 | |
| 				0UL;
 | |
| 
 | |
| 			/* clear phy/link errors */
 | |
| 			scr_write(device, SATA_SCR_ERROR, ~0);
 | |
| 
 | |
| 			/* clear host errors */
 | |
| 			*(sata_regs_base[device] + SATA_CONTROL_OFF) |=
 | |
| 				SATA_SCTL_CLR_ERR;
 | |
| 
 | |
| 			/* clear interrupt register as this clears the error bit in the IDE
 | |
| 			 status register */
 | |
| 			*(sata_regs_base[device] + SATA_INT_CLR_OFF) = ~0UL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return !num_disks_found;
 | |
| }
 | |
| 
 | 
